Вызов метода Python по имени

82

Если у меня есть объект и имя метода в строке, как я могу вызвать этот метод?

class Foo:
    def bar1(self):
        print 1
    def bar2(self):
        print 2

def callMethod(o, name):
    ???

f = Foo()
callMethod(f, "bar1")
Джаз
источник
1
Хотя они похожи, это не точная копия этого вопроса , который спрашивает о функции в модуле, а не о методе объекта.
Grumdrig
Очень тесно связано (черт возьми, может быть, даже обман): как получить доступ к атрибуту объекта с учетом строки, соответствующей имени этого атрибута . Методы тоже атрибуты.
Аран-Фей

Ответы:

113

Воспользуйтесь встроенной getattr()функцией:

class Foo:
    def bar1(self):
        print(1)
    def bar2(self):
        print(2)

def call_method(o, name):
    return getattr(o, name)()


f = Foo()
call_method(f, "bar1")  # prints 1

Вы также можете использовать setattr()для установки атрибутов класса по именам.

Энрико Карлессо
источник
Я не нашел в документации, что искать! Благодаря!
Jazz
@Jazz, это под встроенными функциями. Возможно, вам придется выполнить поиск на странице с помощьюC-f
aaronasterling
@aaronasterling Я знаю, но не могу найти, какое слово искать!
Jazz
почему callMethod (f, "bar1") не называется f.callMethod (f, "bar1")
Филип Путенвила
PhilipJ, потому что это просто метод, определенный несколькими строчками ранее, вне класса Foo :.
Энрико Карлессо
6

У меня был аналогичный вопрос, я хотел вызвать метод экземпляра по ссылке. Вот забавные вещи, которые я нашел:

instance_of_foo=Foo()

method_ref=getattr(Foo, 'bar')
method_ref(instance_of_foo) # instance_of_foo becomes self

instance_method_ref=getattr(instance_of_foo, 'bar')
instance_method_ref() # instance_of_foo already bound into reference

Python потрясающий!

Ярослав Ставничий
источник
2
getattr(globals()['Foo'](), 'bar1')()
getattr(globals()['Foo'](), 'bar2')()

Нет необходимости сначала создавать экземпляр Foo!

Бьёрн
источник
Это был всего лишь пример, у меня есть реальный экземпляр настоящего класса!
Jazz
2
Вызов метода неинициализированного класса может означать, что вы что-то делаете неправильно.
Энрико Карлессо,
что если fooнет в глобальных переменных?
aaronasterling
1
вероятно, это не так, но вы должны позаботиться об Fooэтом. ;)
johndodo 01
2
def callmethod(cls, mtd_name):    
    method = getattr(cls, mtd_name)
    method()
Хтехно
источник
0

Вот более общая версия с использованием декораторов Python. Вы можете позвонить по короткому или длинному имени. Я нашел это полезным при реализации CLI с короткими и длинными подкомандами.

Декораторы Python прекрасны. Брюс Экель (Thinking in Java) прекрасно описывает здесь декораторы Python.

http://www.artima.com/weblogs/viewpost.jsp?thread=240808 http://www.artima.com/weblogs/viewpost.jsp?thread=240845

#!/usr/bin/env python2

from functools import wraps


class CommandInfo(object):
    cmds = []

    def __init__(self, shortname, longname, func):
        self.shortname = shortname
        self.longname = longname
        self.func = func


class CommandDispatch(object):
    def __init__(self, shortname, longname):
        self.shortname = shortname
        self.longname = longname

    def __call__(self, func):
        print("hello from CommandDispatch's __call__")

        @wraps(func)
        def wrapped_func(wself, *args, **kwargs):
            print('hello from wrapped_func, args:{0}, kwargs: {1}'.format(args, kwargs))
            func(wself, *args, **kwargs)

        ci = CommandInfo
        ci.cmds += [ci(shortname=self.shortname, longname=self.longname, func=func)]
        return wrapped_func

    @staticmethod
    def func(name):
        print('hello from CommandDispatch.func')

        for ci in CommandInfo.cmds:
            if ci.shortname == name or ci.longname == name:
                return ci.func

        raise RuntimeError('unknown command')


@CommandDispatch(shortname='co', longname='commit')
def commit(msg):
    print('commit msg: {}'.format(msg))


commit('sample commit msg')         # Normal call by function name

cd = CommandDispatch
short_f = cd.func(name='co')        # Call by shortname
short_f('short sample commit msg')

long_f = cd.func(name='commit')     # Call by longname
long_f('long sample commit msg')


class A(object):
    @CommandDispatch(shortname='Aa', longname='classAmethoda')
    def a(self, msg):
        print('A.a called, msg: {}'.format(msg))


a = A()
short_fA = cd.func(name='Aa')
short_fA(a, 'short A.a msg')

long_fA = cd.func(name='classAmethoda')
long_fA(a, 'short A.a msg')
Нитин Маппаланени
источник