Przekazywanie funkcji z argumentami do innej funkcji w Pythonie?


203

Czy można przekazać funkcje z argumentami do innej funkcji w Pythonie?

Powiedz coś takiego:

def perform(function):
    return function()

Ale funkcje do przekazania będą miały takie argumenty jak:

action1()
action2(p)
action3(p,r)

Odpowiedzi:


289

Masz na myśli to?

def perform( fun, *args ):
    fun( *args )

def action1( args ):
    something

def action2( args ):
    something

perform( action1 )
perform( action2, p )
perform( action3, p, r )

10
Co z nazwanymi parametrami? To znaczy, def action1(arg1, arg2=None, arg3=None)w jaki sposób mógłbyś przekazać argument, że zamierzasz zostać przypisany na przykład do arg3?
ChaimKut,

6
występować (zabawa, ** argumenty), patrz stackoverflow.com/questions/8954746/…
Mannaggia

Co jeśli performi action1, action2na różnych plików? @ S.Lott
alper

@alper zaimportuj je
pfabri

122

Po to jest lambda:

def Perform(f):
    f()

Perform(lambda: Action1())
Perform(lambda: Action2(p))
Perform(lambda: Action3(p, r))

7
Również z ciekawości, czy możesz mi powiedzieć, dlaczego lambdas nie są dobre w tym przypadku?
Joan Venge

11
lambda są jedną z najlepszych cech dobrych języków programowania. niestety implementacja Pythona jest poważnie ograniczona. w tym przypadku jednak pasują idealnie
Javier,

3
Uważam, że ograniczona składnia jest prawie nieprzejrzysta; trudno je wyjaśnić n00bz. Tak, tutaj działają, a mylące funkcje składni są nieobecne. Jest to - być może - jedyny przykład, jaki widziałem lambda, który nie jest niejasny.
S.Lott,

11
Aby móc pobrać wynik przekazanej funkcji, nie byłoby lepiej, gdyby Perform () wywoływał „return f ()”, a nie tylko wywoływał funkcję f ().
mhawke

Myślę, że wersja lambda jest całkiem fajna, ale dziwnie w testach, które przeprowadziłem, wolniej było wywoływać funkcje za pomocą lambda niż metodą fn (* args) omówioną w innej odpowiedzi.
Richard Shepherd

39

Możesz użyć funkcji częściowej z funools takich jak to.

from functools import partial

def perform(f):
    f()

perform(Action1)
perform(partial(Action2, p))
perform(partial(Action3, p, r))

Działa również ze słowami kluczowymi

perform(partial(Action4, param1=p))

1
functools.partialjest również bardziej wszechstronny, jeśli performtrzeba przekazać dalsze parametry f. Np. Można zadzwonić perform(partial(Action3, p))i perform(f)zrobić coś takiego f("this is parameter r").
Robert

13

Używaj funools.partial, a nie lambdów! I Ofc Perform jest bezużyteczną funkcją, możesz przekazywać funkcje bezpośrednio.

for func in [Action1, partial(Action2, p), partial(Action3, p, r)]:
  func()


3
Zależy to od tego, czy argumenty mają być oceniane w witrynie wywoływania funkcji Perform, czy nie.
Dave

6

(miesiące później) mały prawdziwy przykład, w którym lambda jest przydatna, częściowo nie:
powiedz, że chcesz różnych 1-wymiarowych przekrojów poprzez funkcję 2-wymiarową, takich jak przekroje przez rząd wzgórz.
quadf( x, f )bierze 1-d fi nazywa go różnymi x.
Aby nazwać to cięciem pionowym przy y = -1 0 1 i cięciem poziomym przy x = -1 0 1,

fx1 = quadf( x, lambda x: f( x, 1 ))
fx0 = quadf( x, lambda x: f( x, 0 ))
fx_1 = quadf( x, lambda x: f( x, -1 ))
fxy = parabola( y, fx_1, fx0, fx1 )

f_1y = quadf( y, lambda y: f( -1, y ))
f0y = quadf( y, lambda y: f( 0, y ))
f1y = quadf( y, lambda y: f( 1, y ))
fyx = parabola( x, f_1y, f0y, f1y )

O ile mi wiadomo, partialnie mogę tego zrobić -

quadf( y, partial( f, x=1 ))
TypeError: f() got multiple values for keyword argument 'x'

(Jak dodać do tego tagi numpy, częściowy, lambda?)


5

Nazywa się to funkcjami częściowymi i można to zrobić na co najmniej 3 sposoby. Moim ulubionym sposobem jest użycie lambda, ponieważ pozwala uniknąć zależności od dodatkowego pakietu i jest najmniej gadatliwy. Załóżmy, że masz funkcję add(x, y)i chcesz przejść add(3, y)do innej funkcji jako parametru, tak aby inna funkcja decydowała o wartości y.

Użyj lambda

# generic function takes op and its argument
def runOp(op, val):
    return op(val)

# declare full function
def add(x, y):
    return x+y

# run example
def main():
    f = lambda y: add(3, y)
    result = runOp(f, 1) # is 4

Stwórz własne opakowanie

Tutaj musisz utworzyć funkcję, która zwraca funkcję częściową. To oczywiście dużo bardziej szczegółowe.

# generic function takes op and its argument
def runOp(op, val):
    return op(val)

# declare full function
def add(x, y):
    return x+y

# declare partial function
def addPartial(x):
    def _wrapper(y):
        return add(x, y)
    return _wrapper

# run example
def main():
    f = addPartial(3)
    result = runOp(f, 1) # is 4

Użyj częściowego z Funkools

Jest to prawie identyczne jak lambdapokazano powyżej. Dlaczego więc tego potrzebujemy? Jest kilka powodów . Krótko mówiąc, partialw niektórych przypadkach może być nieco szybszy (zobacz jego implementację ) i można go użyć do wczesnego wiązania w porównaniu do późnego wiązania lambda.

from functools import partial

# generic function takes op and its argument
def runOp(op, val):
    return op(val)

# declare full function
def add(x, y):
    return x+y

# run example
def main():
    f = partial(add, 3)
    result = runOp(f, 1) # is 4

1

Oto sposób na to z zamknięciem:

    def generate_add_mult_func(func):
        def function_generator(x):
            return reduce(func,range(1,x))
        return function_generator

    def add(x,y):
        return x+y

    def mult(x,y):
        return x*y

    adding=generate_add_mult_func(add)
    multiplying=generate_add_mult_func(mult)

    print adding(10)
    print multiplying(10)

W każdym razie należy zrobić coś więcej niż tylko przekazywanie funkcji do innego zamknięcia.
jake77
Korzystając z naszej strony potwierdzasz, że przeczytałeś(-aś) i rozumiesz nasze zasady używania plików cookie i zasady ochrony prywatności.
Licensed under cc by-sa 3.0 with attribution required.