Lua中优雅的运行三目运算符


1、引言

  在其他语言中我们都提供了形如:a ? b : c 的逻辑,如:在C或者C++中我们常常会用到上面的三目运算符(或者叫三元运算符),这句话的简单解释就是如果a为真,表达式的值是b,否者表达式的值是c。这样的逻辑在写lua的时候也常常到,所以有人发明了a and b or c这种写法,下面来分析一下这个表达式有什么问题,并给出解决办法。

2、理论普及

  这里引用lua的经典书籍《Lua程序设计》这本书中的内容可知:Lua中的逻辑操作符有and、or和not。与条件控制语句一样,所有的逻辑操作符将false和nil视为假,而将其他的任何东西视为真,对于操作符and来说,如果它的第一个操作数为假,就返回第一个操作数;不然返回第二个操作数。对于操作符or来说,如果它的第一个操作数为真,就返回第一个操作数;不然返回第二个操作数。
  引用lua的经典书籍《Lua程序设计》这本书中的内容可知:Lua中的逻辑操作符有and、or和not。与条件控制语句一样,所有的逻辑操作符将false和nil视为假,而将其他的任何东西视为真,对于操作符and来说,如果它的第一个操作数为假,就返回第一个操作数;不然返回第二个操作数。对于操作符or来说,如果它的第一个操作数为真,就返回第一个操作数;不然返回第二个操作数。

  • and
local a = 表达式1 and 表达式2

理解:
  如果表达式1的结果为假则不执行表达式2直接返回表达式1的结果,则整个表达式的结果就是表达式1的结果,a的值就是表达式1的值(即为:假);如果表达式1的结果为真,继续执行表达式2,返回表达式2的结果,整个表达式的结果就是表达式2的结果,a的值是表达式2的结果。

  • or
local a = 表达式1 or 表达式2

理解:
  如果表达式1的结果为真则不执行表达式2直接返回表达式1的结果,则整个表达式的结果就是表达式1的结果,a的值就是表达式1的值(即为:真);如果表达式1的结果为假,继续执行表达式2,返回表达式2的结果,整个表达式的结果就是表达式2的结果,这里a的值是表达式2的结果。

  • not
local a = not 表达式1

理解:
  not就是取反,a的值就是表达式1的值取反。

3、加深

  书中还有一段写到:and和or都是用“短路求值”,也就是说,它们只会在需要时才会去评估第二个操作数。短路求值可以确保像(type(v)==”table” and v.tag == “best”)这样的表达式不会导致运行错误,这里的“短路求值”可以保证在v不是table的情况,不会判断v.tag == “best”的值而直接返回false。

4、提出问题

  另外书中还提到“a and b or c“是一种类似于C语言中的表达式a ? b : c的习惯写法,可是并未指出其中的问题,但是译者在此注明了观点,他指出如果要想让a and b or c等价于a ? b : c,前提是b表达式必须为真,也就是说b不能等于false或者nil,最后译者给出了建议,那就是在无法确认b为真的情况下,最常用的办法还是使用正常的if-else语句。但是我们习惯了写三元表达式,感觉很方便,不是吗?下面我们来一起分析一下。

5、分析问题

  说到这里可能有些人开始反迷糊,为什么b表达式的值必须为真呢?我们不妨写个例子:

local a = false
local b = false
local c = 20

local res = a and b or c
print(res)  --> 20
local a = true
local b = false
local c = 20

local res = a and b or c
print(res)  --> 20

  通过以上栗子(换为nil也是一样这里不贴代码了)尝试,如果b == false,那么无论a的表达式为真或者假,整体都会返回c的值。那么怎样才能保证b的值一直为真呢?实际的程序逻辑中是允许b为false或者nil的啊!

6、解决办法

  这就用到了下面的技巧,话说这也是我无意间从其他人那里学到的,写到这里也是为了记录一下,那就是在b表达式的外边包装一层table,写成{b}的形式,返回值的时候在写成{b}[1]的形式就好了,那么整体的表达式就变成了(a and {b} or {c})[1],这样就满足了第二个表达式恒为真的前提,今后你可以在lua中悦快地写三目运算符了!

7、结论

  • 完美运行山姆运算符解决方案,形如
(a and {b} or {c})[1]

The End
  好了,今天的分享就到这里,如有不足之处,还望大家及时指正,随时欢迎大家留言评论和探讨交流!!!

猜你喜欢

转载自blog.csdn.net/lxt610/article/details/80816637
今日推荐