您不想为每首歌曲创建一个类;您想为每首歌曲创建一个类的实例。例如:
class Song(object):
def __init__(self, name):
self.name = name
songNames = ['song1', 'song2', 'song3']
songs = []
for name in songNames:
songs.append(Song(name))
请注意,我将Song
对象存储在列表中。您需要将它们存储在某个地方,或者以后无法再次访问它们,对吗?
如果您希望以后能够按名称访问它们,而不是仅仅遍历所有它们:
songs = {}
for name in songNames:
songs[name] = Song(name)
您可以使用推导将其中任何一个转换为单行:
songs = [Song(name) for name in songNames]
songs = {name: Song(name) for name in songNames}
来自评论:
我原本打算按班级看起来像下面这样,因为每首歌曲都将具有所有相同的变量:
class songData:
title = ""
artist = ""
directory = ""
tempStr = ""
def description(self):
desc_str = "%s is by %s, located at %s" %(self.title, self.artist,self.directory)
return desc_str
由于我已经解释过的相同原因,这是错误的:您正在创建类属性,而不是实例属性。这意味着每个实例都将共享一个标题等,这不是您想要的。此外,这意味着self.title
具有误导性(尽管它会起作用)。
同时,如果这是 Python 2,那么您将再次创建一个经典类。此外,您可能需要一种__str__
方法而不是特殊description
方法,因为那样您就可以print(song)
而不是必须print(song.description())
. 最后,对变量和类使用相同的命名约定有点令人困惑。
你可能想要的是:
class SongData:
def __init__(self, title="", artist="", directory=""):
self.title, self.artist, self.directory = title, artist, directory
def __str__(self):
desc_str = "%s is by %s, located at %s" % (self.title, self.artist,self.directory)
return desc_str
现在,您可以像上面的类一样使用它Song
:
songNames = ['song1', 'song2', 'song3']
songs = {name: SongData(title=name) for name in songNames}
现在您可以执行以下操作:
name = input('What song do you want info on?') # raw_input, if Python 2
print(songs[name])
(当然这不是很有用,除非你也有在某处设置艺术家和目录的代码,因为它只会 print Song Windowlicker is by , located at
。但我不知道你打算从哪里得到这些。)
您的新代码有两个问题。第一的:
def __str__(self):
desc_str = "Title: %s \nArtist: %s \nDirectory: %s \n" %(self.title,
self.artist, self.directory)
print desc_str
你需要在return desc_str
这里,不是print
它。
第二:
songNames = ['song1', 'song2', 'song3']
artistNames = ['artist1', 'artist2', 'artist3']
dirNames = ['dir1', 'dir2', 'dir3']
songs = {name: SongData(title=name) for name in songNames}
artists = {band: SongData(artist=band) for band in artistNames}
directorys = {direc: SongData(directory=direc) for direc in dirNames}
在这里,您将创建三个单独的SongData
对象集合,每个对象只填充一个属性。
这里的关键是zip
。一旦你了解了它的功能,它就是 Python 中最有用的函数之一,但在你了解它之前,你永远不会想到去寻找它。如果我展示它的作用,则更容易描述:
>>> zip(songNames, artistNames, dirNames)
[('song1', 'artist1', 'dir1'),
('song2', 'artist2', 'dir2'),
('song3', 'artist3', 'dir3')]
因此,这会为您提供一个元组列表,其中每个元组都有一个歌曲名称、一个艺术家和一个目录。第一个元组是每个元组的第一个,第二个是每个元组的第二个,依此类推。
现在你可以SongData
很容易地从每个元组中构建一个:
songs = {}
for title, artist, directory in zip(songNames, artistNames, dirNames):
songs[title] = SongData(title, artist, directory)
作为一个 dict 理解,它变得有点冗长:
songs = {title: SongData(title, artist, directory)
for title, artist, directory in zip(songNames, artistNames, dirNames)}
但是你可以用另一个技巧来简化它:解包参数:
songs = {t[0]: SongData(*t) for songtuple in zip(songNames, artistNames, dirNames)}
当然,你可以不这样做zip
,但它看起来像一团糟:
songs = {SongData(songNames[i], artistNames[i], dirNames[i])
for i in range(len(songNames))}
......如果你有一个列表不匹配的小错误,那么这种方式将很难理解和调试。通常,每当您使用 Python 编写for i in range(len(foo))
代码时,都可能有一种简单的方法。
无论您如何构建它,您都可以按照您的预期使用它:
>>> print songs['song1']
Title: song1
Artist: artist1
Directory: directory1
当我们这样做时,您可能不希望在每行输出的末尾有一个空格。它对人类是不可见的,但它可能会混淆您以后编写的用于解析输出的代码,并且它会浪费空间,并且没有真正的好处。只需将每个\n
放在%s
.