Интересен въпрос. Ето как бих постъпил.
(Това работи в python2. Не съм го тествал в python3, но няма да се изненадам, ако не работи...)
Можете да итерирате всички „номинирани“, като използвате reversed(inspect.getmro(cls))
и връщате първия (чрез извличане на next
стойността на итератора), който отговаря на условието, че има съответното attr
и че attr
е същото като метода на съответното cls
.
Сравнението на идентичността на метода се извършва чрез сравняване на атрибута im_func
на необвързания метод.
import inspect
def getMethodClass(cls, attr):
return next(
basecls for basecls in reversed(inspect.getmro(cls))
if hasattr(basecls, attr)
and getattr(basecls, attr).im_func is getattr(cls, attr).im_func
)
getMethodClass(A, 'a')
=> __main__.A
getMethodClass(B, 'a')
=> __main__.A
getMethodClass(B, 'b')
=> __main__.B
# an alternative implementation, suggested by @chameleon
def getAttributeClass(cls, attrName):
# check first if has attribute
attr = getattr(cls, attrName)
mro = inspect.getmro(cls)
# only one class on list
if len(mro) == 1:
return cls
# many class on list
for base in reversed(mro[1:]):
# check if defined in this base
try:
baseAttr = getattr(base, attrName)
except AttributeError:
continue
else:
if baseAttr.im_func is attr.im_func:
return base
# define in top class
return cls
Функцията може да има и сигнатурата, която предлагате:
def getMethodClass(unbound_method):
cls = unbound_method.im_class
attr = unbound_method.__name__
# rest of implementation is the same as before...
getMethodClass(B.a)
=> __main__.A
person
shx2
schedule
06.04.2014
getattr(A, 'a')
е просто глупав начин да напишетеA.a
, аtype(A)
е израз на Python, който се оценява наtype
- вероятно искате самоA
. - person   schedule 06.04.2014