10

我正在开发一个基于文本的冒险游戏,并且有一个包含所有动作类的模块,Move(Action)例如Look(Action). 我需要一种将模块中作为类的子类的所有类实例Action化为如下列表的方法:

actions = [Move(), Look()]

有没有一种方法可以做到这一点,而不必通过键入它们的名称来单独实例化这些类?

4

2 回答 2

8

解决方案

这有效:

class Action:
    pass

class Move(Action):
    pass

class Look(Action):
    pass


actions = []
global_objs = list(globals().items())

for name, obj in global_objs:
    if obj is not Action and isinstance(obj, type) and issubclass(obj, Action):
        actions.append(obj())
print(actions)

印刷:

[<__main__.Move object at 0x101916ba8>, <__main__.Look object at 0x101916be0>]

脚步

首先,我们在当前模块中获取所有名称一个对象:

global_objs = list(globals().items())

我们需要转换成一个列表,因为在 Python 3 中 items()返回一个 dictview 对象,它反映了底层字典的变化。由于我们在同一个模块中工作,定义新对象会改变这个字典。转换成列表可以解决这个问题。

接下来,我们遍历所有对象。他们需要满足三个条件才能成为 的子类Action

  1. obj is not Action- 因为Action是它自己的一个子类,我们需要过滤掉它。

  2. isinstance(obj, type)- 对象必须是一个类。否则,将无法进行 3. 下的测试。

  3. issubclass(obj, Action)- 最后,我们可以进行子类测试。

现在我们可以创建所有过滤类的实例并将它们附加到我们的列表中:

actions.append(obj())

较短的版本

如果您确定所有类都定义在一个模块中,或者您甚至希望所有子类分布在多个模块中,那么这会更短:

>>> actions = [obj() for obj in Action.__subclasses__()]
>>> actions
[<__main__.Move at 0x10fc14fd0>, <__main__.Look at 0x10fc14668>]
于 2016-01-01T09:15:33.947 回答
1
from other_module import Action

def is_action_class(var):
    return var != Action and type(var) == type and issubclass(var, Action)

classes = []    

for attr in dir(my_module):
    var = getattr(my_module, attr)
    if is_action_class(var):
        classes.append(var)
instances = [c() for c in classes]
于 2016-01-01T09:15:08.560 回答