Я хочу знать, как вызывать функцию, передавая параметр, который может не ожидать.
Я столкнулся с этой проблемой несколько дней назад и нашел способ ее обойти. Но сегодня я решил, что посмотрю, возможно ли то, что я хотел сделать. К сожалению, я не помню контекст, в котором я его использовал. Так что это глупый пример, в котором есть много лучших способов сделать это, но просто игнорируйте это:
def test(func, arg1, arg2):
return func(arg1, arg2, flag1=True, flag2=False) #Only pass the flags if the function accepts them.
def func1(a, b, flag1, flag2):
ret = a
if flag1:
ret *= b
if flag2:
ret += b
return ret
def func2(a, b):
return a*b
print test(func1, 5, 6) #prints 30
Альтернатива, которую я придумал, выглядела так:
def test(func, arg1, arg2):
numArgs = len(inspect.getargspec(func).args)
if numArgs >= 4:
return func(arg1, arg2, True, False)
elif numArgs == 3:
return func(arg1, arg2, True)
else:
return func(arg1, arg2)
print test(func2, 5, 6) #prints 30
или try..except .. block
Но должен быть лучший способ сделать это без изменения func1 и func2, верно?
(Изменить): Используя решение, предоставленное Max S, я думаю, что это лучший подход:
def callWithOptionalArgs(func, *args):
argSpec = inspect.getargspec(func)
lenArgSpec = len(argSpec.args or ())
argsToPass = args[:lenArgSpec] #too many args
defaults = argSpec.defaults or ()
lenDefaults = len(defaults)
argsToPass += (None, )*(lenArgSpec-len(argsToPass)-lenDefaults) #too few args
argsToPass += defaults[len(argsToPass)+len(defaults)-lenArgSpec:] #default args
return func(*argsToPass)
print callWithOptionalArgs(func1, 5, 6, True) #prints 30
print callWithOptionalArgs(func2, 5, 6, True) #prints 30