【发布时间】:2011-11-07 06:51:25
【问题描述】:
可能重复:
Calling a function from a string with the function's name in Python
我想我可以编写一些糟糕的代码来做到这一点,但我更愿意看到“干净的版本”。
对我来说,最好的方法是制作一个包含给定对象可以使用的各种功能的字典。然后当用户被指示告诉对象它在做什么时,它会根据该字典吐出一个菜单。
我搜索了一下,并没有真正看到适用于我的东西,所以我想我会试一试。好吧,它没有用。
class Man(object):
def __init__(self):
self.cmds = ['foo', 'bar']
def foo(self):
print "Foo called."
def bar(self):
print "Bar called."
def junk(self):
print "Junk called." ##not in dict, on purpose, will explain
def menu(self):
while True:
print "List of actions:"
for acts in self.cmds:
print acts
cmd = raw_input("> ")
if cmd in self.cmds:
cmd() ##doesn't work.
##neither did self.cmd() (got AttributeError, obviously)
result = getattr(self, cmd)() ## this works! thanks cdhowie
else:
pass
Stick = Man()
Stick.menu()
如果不是很明显,每当我输入 if-else 认为是 True 的内容时,程序都会给出 TypeError - 在这种情况下,输入“foo”或“bar”。 事情是这样的,我知道我可以在这里写一个又长又丑的 if-else 东西并使这个例子工作 - 但我希望能够从 self.cmds 中追加/删除来改变对象的功能。因此第三个函数 Junk(); Stick 无法从当前的 dict-menu 访问 'Junk()',但通过一点 self.cmds.append 操作我希望它能够访问。
怪异的 Python,它们是如何工作的?这是解决这个问题的正确方法,还是有更简单的方法?
编辑:我的答案是在 getattr 的魔力中找到的。谢谢 cdhowie。诀窍是更改 while 循环以具有此位: result = getattr(self, cmd)()
我现在知道我的下一个任务是最终弄清楚 getattr() 的实际作用。原谅我的菜鸟身份,呵呵,我不知道我的代码是什么:)
最终编辑:虽然 cdhowie 的示例适用于原始程序,但我后来发现 ders 的回答允许我在功能上做一些我无法用 getattr() 做的事情; ders 的解决方案使我更容易在 Man 的 init 中的其他对象中使用函数 - 我认为这称为“对象组合”对吗?无论如何,getattr() 将 AttributeError 任何从除 Man 之外的任何地方添加到 self.cmds 的函数。或者我可能只是再做一次奇怪的事情。但我只想说,ders FTW。
【问题讨论】:
-
如果您想按名称调用方法,This question 是相关的。
-
哦,那确实可以解决问题。快速简单的回答,谢谢薄荷糖。
-
您是否通过解决您的问题的代码更新来编辑您的问题?如果是这样,这个Q需要被删除或关闭。
-
是的,我同意。由于我显然是新用户,因此我无法再回答自己的问题...... 7 小时。所以是的。
标签: python dictionary menu