【问题标题】:How to pass functions as arguments TO a lambda?如何将函数作为参数传递给 lambda?
【发布时间】:2023-10-31 23:34:01
【问题描述】:

我正在尝试使用 lambda 作为一种方式来处理 GPIO 按钮的上升/下降状态。因此,我传递了下降 (c1) 和上升 (c2) 的回调,以创建稍后按下 GPIO 按钮时将使用的 lambda。但它不起作用:

# Create lambda to handle both rising and falling triggers/callbacks
def trigger_both( self, pin, c1, c2):
    return lambda pin, c1, c2: c1() if GPIO.input(pin) else c2()

c1 和 c2 都是函数。执行 lambda 时运行时出错:

TypeError: () 缺少 2 个必需的位置参数:'c1' 和 'c2' TypeError: () 缺少 2 个必需的位置参数:'c1' 和 'c2'

在上下文中:

    if callback and callback2:
        callback_both = self.trigger_both(pin, callback, callback2)
        GPIO.add_event_detect(pin, GPIO.BOTH, callback=callback_both, bouncetime=200)

【问题讨论】:

  • 你必须想出另一个设计。传递给 add_event_detectcallback 函数仅使用 channel (pin) 参数调用
  • 是的,没错,但我不需要/不想在运行时调用 c1/c2 - 这是我的编码错误。接受的答案修复了它:return lambda pin: c1(pin) if GPIO.input(pin) else c2(pin)

标签: python python-3.x lambda callback


【解决方案1】:

假设 callback_both() 传递了单个参数 pin,我们可能会使用一个简单的 闭包

def trigger_both(self, c1, c2):
    return lambda pin: c1(pin) if GPIO.input(pin) else c2(pin)

如果不是,请考虑更长的形式:

def trigger_both(self, c1, c2):
    return lambda pin, f1=c1, f2=c2: f1(pin) if GPIO.input(pin) else f2(pin)

如果由事件处理程序传递,则无需将pin 传递给trigger_both()。我只是将pin 传递给c1()c2() 以确保完整性——您可以删除或忽略它。我们也可以用partial 代替lambda 来做到这一点:

from functools import partial

def trigger_both(self, c1, c2):
    def callback_both(f1, f2, pin):
        return f1(pin) if GPIO.input(pin) else f2(pin)

    return partial(callback_both, c1, c2)

或者,还是一个更简单的闭包:

def trigger_both(self, c1, c2):
    def callback_both(pin):
        return c1(pin) if GPIO.input(pin) else c2(pin)

    return callback_both

【讨论】: