1

我基本上是在尝试从播放列表中提取歌曲名称,然后我想为播放列表中的每首歌曲创建一个类,其中包括该歌曲的所有相关数据(歌曲标题、艺术家、每首歌曲的位置等)

我有一个列表,其中列表中的每个项目都是歌曲名称。我正在尝试创建一个循环,该循环创建一个以列表中每个项目命名的新类。

这是我到目前为止所拥有的:

class MyClass():
    var = "hi"

songNames = ['song1', 'song2', 'song3']

for name in songNames:
    name = MyClass()

这不起作用,我认为这是因为 Python 遇到问题/无法为这样的类分配名称。我是 Python 的初学者,经过大量搜索后,我无法提出解决方案。这样做的正确方法是什么?


所以我一直在研究代码并取得了一些进展:

class SongData(object):
    def __init__(self, title="", artist="", directory=""):
        self.title, self.artist, self.directory = title, artist, directory
    def __str__(self):
        desc_str = "Title: %s \nArtist: %s \nDirectory: %s \n" %(self.title,
                                            self.artist, self.directory)
        print desc_str

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}

我希望能够打印出desc_str每首歌曲的内容,如下所示:

Title: song1 
Artist: artist1 
Directory: dir1

但到目前为止,我只设法一次调用一个数据类别,因此它会打印Title: song1但将Artist:andDirectory:部分留空。

我怎样才能让它一次打印出来?

4

2 回答 2

15

您不想为每首歌曲创建一个类;您想为每首歌曲创建一个类的实例。例如:

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.

于 2013-07-15T19:27:12.127 回答
2

我同意您似乎想为每首歌曲创建一个实例而不是一个类。但是,如果您确实想为每首歌曲创建一个类,请使用该type(className, parentTuple, attributeDict)函数。它返回一个具有给定名称的新类。

例如

songClasses = []
for name in songNames:
    songClasses.append(type(name, (), {}))
于 2013-07-15T19:30:20.817 回答