Passare funzioni con argomenti a un'altra funzione in Python?


205

È possibile passare funzioni con argomenti a un'altra funzione in Python?

Dì qualcosa come:

def perform(function):
    return function()

Ma le funzioni da passare avranno argomenti come:

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

Risposte:


291

Intendi questo?

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

def action1( args ):
    something

def action2( args ):
    something

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

10
Che dire dei parametri nominati? Cioè def action1(arg1, arg2=None, arg3=None), come potresti passare un argomento che intendi assegnare ad arg3, per esempio?
ChaimKut,

6
perform (fun, ** args), vedi stackoverflow.com/questions/8954746/…
Mannaggia,

E se performe action1, action2su file diversi? @ S. Lott
Alper

@alper importare loro
pfabri

122

Questo è lambda per:

def Perform(f):
    f()

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

7
Anche per curiosità, puoi dirmi perché le lambda non vanno bene in questo caso?
Joan Venge,

11
I lambda sono una delle migliori caratteristiche di buoni linguaggi di programmazione. sfortunatamente, l'implementazione di Python è fortemente limitata. in questo caso, tuttavia, si adattano perfettamente
Javier,

3
Trovo che la sintassi limitata sia quasi opaca; sono difficili da spiegare a n00bz. Sì, funzionano qui e le funzioni confuse della sintassi sono assenti. Questo è - forse - l'unico esempio che ho visto di una lambda che non è oscuro.
S.Lott

11
In modo che tu possa recuperare il risultato della funzione passata, non sarebbe meglio se Perform () chiamasse "return f ()" anziché chiamare semplicemente f ().
mhawke,

Penso che la versione lambda sia abbastanza ordinata, ma stranamente nei test che ho eseguito è stato più lento chiamare funzioni tramite lambda rispetto al metodo fn (* args) discusso in un'altra risposta.
Richard Shepherd,

39

È possibile utilizzare la funzione parziale da functools in questo modo.

from functools import partial

def perform(f):
    f()

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

Funziona anche con parole chiave

perform(partial(Action4, param1=p))

1
functools.partialè anche più versatile se è performnecessario consegnare ulteriori parametri a f. Ad esempio, si potrebbe chiamare perform(partial(Action3, p))e perform(f)fare qualcosa del genere f("this is parameter r").
Robert,

13

Usa functools.partial, non lambdas! E ofc Perform è una funzione inutile, puoi passare direttamente alle funzioni.

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


3
Dipende se si desidera che gli argomenti vengano valutati nel sito di chiamata di Perform oppure no.
Dave,

6

(mesi dopo) un piccolo esempio reale in cui lambda è utile, parziale no:
supponiamo che tu voglia varie sezioni trasversali monodimensionali attraverso una funzione bidimensionale, come fette attraverso una fila di colline.
quadf( x, f )prende un 1-d fe lo chiama per vari x.
Per chiamarlo per tagli verticali su y = -1 0 1 e tagli orizzontali su 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 )

Per quanto ne so, partialnon posso farlo -

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

(Come aggiungere tag numpy, parziale, lambda a questo?)


5

Questo si chiama funzioni parziali e ci sono almeno 3 modi per farlo. Il mio modo preferito è usare lambda perché evita la dipendenza da pacchetti extra ed è il meno dettagliato. Supponiamo di avere una funzione add(x, y)e di voler passare add(3, y)ad un'altra funzione come parametro in modo che l'altra funzione decida il valore pery .

Usa 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

Crea il tuo wrapper

Qui è necessario creare una funzione che restituisca la funzione parziale. Questo è ovviamente molto più dettagliato.

# 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

Usa parziale da functools

Questo è quasi identico a quello lambdamostrato sopra. Allora perché ne abbiamo bisogno? Ci sono alcuni motivi . In breve, partialin alcuni casi potrebbe essere un po 'più veloce (vedi la sua implementazione ) e puoi usarlo per il binding anticipato rispetto al binding tardivo 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

Ecco un modo per farlo con una chiusura:

    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)

In ogni caso, è necessario fare di più che passare una funzione a un'altra chiusura.
jake77
Utilizzando il nostro sito, riconosci di aver letto e compreso le nostre Informativa sui cookie e Informativa sulla privacy.
Licensed under cc by-sa 3.0 with attribution required.