在进行自动化的时候写了这么一个语句
check_online = lambda x: x is not None and x.get("is_online") == '1' if online_status else lambda x: x is not None and x.get("is_online") != '1'
这条语句的目的大概是根据期望状态的不一样,希望返回不一样的检查函数,这里的检查函数使用匿名函数也就是lambda表达式来写,实际测试的时候呢发现,当检查状态为True时即online_status为真时,返回没有问题,但是当检查状态为False时,就发现了检查函数会出错,本来期望当设备离线时
x.get("is_online") != '1'
才会通过检查,结果为“在线状态的时候”x.get("is_online") == '1')
就通过了检查,这是怎么回事呢?
于是我写了这样一串代码来测试
if not online_status:
self.logger.debug(f"当前想判断是否{online_status},校验结果为{check_online(device_info)},设备状态为{device_info.get('is_online')}")
打上断点测试一下,输出如下
DEBUG: 当前想判断是否False,校验结果为<function DeviceCheck.wait_until_device_online.<locals>.<lambda>.<locals>.<lambda> at 0x000001CA0DA3B6A8>,设备状态为1
果然有猫腻,这里的检验结果居然是一个函数,大家都知道,python里函数也是对象,而对于非空对象,布尔判断为True
,所以这里应该是因为返回了一个函数对象所以一直为真通过了检查,正确的写法是
check_online = lambda x: x is not None and x.get("is_online") == '1' if online_status else x is not None and x.get("is_online") != '1'
这也说明了lamda的默认作用域包含整个冒号后面的语句