>我有一个带有两个函数的导入模块
def kind1(paramA, paramB)
...
def kind2(paramA, paramB)
我通过构造函数将这两个函数中的一个随机传递给对象。
object = Constructor(param1, param2, function)
现在有了
>>> print object.function
我得到
function kind1 at 0xb124...
或
function kind2 at 0xb1464...
如何检查对象函数是 kind1 还是 kind2?
我试过
if object.function == kind1:
print 'kind1'
elif object.function == kind2:
print 'kind2'
并与
if object.function == kind1():
print 'kind1'
elif object.function == kind2():
print 'kind2'
我也试过
if isinstance(object.function, kind1):
print 'kind1'
elif isinstance(object.function, kind2):
print 'kind2'
但他们显然没有工作。执行此检查的正确方法是什么?
我认为您需要将函数(方法)称为ObjectInstance.kind1
,并ClassOfObject.kind1
因为 type(ClassOfObject.kind1) == <class 'function'>
和 type(ObjectInstance.kind1) == <class 'method'>
和具有不同的地址很难直接比较,解决方法是使用一些属性来比较它(例如.__name__
或创建自己的属性)
您的代码示例与您描述的行为不匹配,将函数对象分配给属性不会更改该对象的标识:
def foo():
pass
class Klass(object):
def __init__(self, f):
self.f = f
a = Klass(foo)
print(a.f is foo) # True
print(a.f == foo) # True
使用 id
函数,该函数检查对象标识:
def foo():
pass
def bar():
pass
def whatisit(fn):
case = {
id(foo) : "foo",
id(bar) : "bar",
}
print "got function %s" % case[id(fn)]
# see what we get
whatisit(foo)
whatisit(bar)
=>
got function foo
got function bar
请注意,使用 fn.__name__
可能是可以接受的,也可能是不可接受的,具体取决于您的特定系统(在某些系统中,这将被视为安全风险):
def foo():
pass
def sybil():
pass
# let's be evil
sybil.__name__ = 'foo'
# social engineering ;-)
print foo.__name__ == sybil.__name__
#=> True
# always check identity
print id(foo) == id(sybil)
#=> False
如果必须依赖名称而不是 id(例如,如果将函数传递给子进程或远程进程),则至少在只读代码对象中使用安全路径:
# the unforgeable name, similar to a fingerprint
print foo.__code__.co_name == sybil.__code__.co_name
#=> False
如果您想知道恶意功能的花哨名称,请查看维基百科的爱丽丝和鲍勃。