今天遇到一个关于php 嵌套使用条件运算符(ternary expressions)的问题
现象先来看一段c语言代码(test.c):
#include<stdio.h>int main() { int x = 1; int shit = x == 1 ? 100 : x == 2 ? 200 : 300; printf(shit的值:%d\n, shit); return 0;}
编译后运行一下
root$ gcc test.c -o test && ./testshit的值:100
答案在意料之中,因为x==1,所以100被赋值给shit。
但是如果我们用php重写一下上文的代码(test.php):
执行一下:
root$ php test.phpshit的值:200
我们发现返回的结果不一样了,这是为什么呢?
排查首先怀疑可能是php中比较运算符(==)和条件运算符(?:)的优先级问题,我们查看一下php官方文档
==的优先级比?:更高(c语言也是这样),所以
$shit = $x == 1 ? 100 : $x == 2 ? 200 : 300;
等效于
$shit = ($x == 1) ? 100 : ($x == 2) ? 200 : 300;
执行一遍也确实如此,可以排除掉是运算符优先级导致问题的可能性了。
但是官方文档里关于运算符结合方向的举例说明中出现了这么一句话:
这跟上文描述的现象很相似,问题应该就在这了。一番查阅之后得到以下结论:
结论c语言的条件运算符(?:)的结合方向是从右往左,每次求值都是从最右边的子表达式开始算起,所以
int x = 1;int shit = x == 1 ? 100 : x == 2 ? 200 : 300;//等效于int shit = x == 1 ? 100 : (x == 2 ? 200 : 300);//等效于int shit = x == 1 ? 100 : (300);// 100
php的条件运算符(?:)的结合方向是从左往右,每次求值都是从最左边的子表达式开始算起,所以
$x = 1;$shit = $x == 1 ? 100 : $x == 2 ? 200 : 300;//等效于$shit = ($x == 1 ? 100 : $x == 2) ? 200 : 300;//等效于$shit = (100) ? 200 : 300;// 200
介于php的条件运算符结合方向,我们无法像c/c++那样 通过嵌套条件运算符来达到if-elseif-elseif-else表达式的效果,除非我们在靠后的子表达式中加上括号,本例中就可以靠这种方式解决:
$shit = $x == 1 ? 100 : ($x == 2 ? 200 : 300);
但在条件分支较多的情况下,就会出现代码可读性问题(堆积括号):
$shit = $x == 1 ? 100 : ($x == 2 ? 200 : ($x== 3 ? 300 : ... ($x == 8 ? 800 : 900)))))));
由于php不堆积括号的写法与c/c++在执行结果上是不一致的,并且只能通过加括号改变默认的结合方向 以达到预期的结果,所以php文档里干脆不建议嵌套使用条件运算符:
note:
it is recommended that you avoid stacking ternary expressions. php's
behaviour when using more than one ternary operator within a single statement is non-obvious以上就是本文的全部内容,希望对大家的学习有所帮助,更多相关内容请关注!
相关推荐:
聊聊框架开发的依赖注入,容器与外观模式(下部)
如何解决php的高并发和大流量的问题
以上就是关于php条件运算符遇到的一个问题及解决方法的详细内容。
