所以,我刚刚阅读this blog post,我感到困惑的是"三元运算符是左关联的"部分,所以我在解释器中运行示例代码:
$arg = 'T';
$vehicle = ( ( $arg == 'B' ) ? 'bus' :
( $arg == 'A' ) ? 'airplane' :
( $arg == 'T' ) ? 'train' :
( $arg == 'C' ) ? 'car' :
( $arg == 'H' ) ? 'horse' :
'feet' );
echo $vehicle;
实际上,它返回horse
,这是博客文章中的反直觉。
出于好奇,我接着试着做这项工作"通过重新编写它来符合我的想法" left-associative"会想要的。我得到了这个(格式化很奇怪,但至少在我的脑海中它更清晰):
$arg = 'T';
$vehicle = ( ( $arg != 'B' ) ?
( $arg != 'A' ) ?
( $arg != 'T' ) ?
( $arg != 'C' ) ?
( $arg != 'H' ) ?
'feet' :
'horse' :
'car' :
'train' :
'airplane' :
'bus'
);
echo $vehicle;
现在,这可以正常运作,因为任何字符$arg
都会返回以该字符开头的车辆(当然是小写,但这里并不重要)。
好奇,因为我不清楚为什么前者不起作用,我想知道后者是否会以右联语言失败,因为它似乎不会。所以我在java解释器中测试了它。这里的代码适用于任何想要尝试并节省几秒钟的人。
class Main {
public static void main(String[] args) {
Character arg = 'a';
String vehicle = ( ( arg == 'B' ) ? "bus" :
( arg == 'A' ) ? "airplane" :
( arg == 'T' ) ? "train" :
( arg == 'C' ) ? "car" :
( arg == 'H' ) ? "horse" :
"feet" );
System.out.println(vehicle);
vehicle = ( ( arg != 'B' ) ?
( arg != 'A' ) ?
( arg != 'T' ) ?
( arg != 'C' ) ?
( arg != 'H' ) ?
"feet" :
"horse" :
"car" :
"train" :
"airplane" :
"bus"
);
System.out.println(vehicle);
}
}
这两种格式都适用于java。那么,什么给PHP?我听说它主要是评估最终的平等($arg == 'H'
)。但如果是这样,那就意味着它表现得好像是
$vehicle = ((($arg == 'B') || ($arg == 'A') || ($arg == 'T') ||
($arg == 'C') || ($arg == 'H')) ? 'horse' : 'feet');
这对我没有意义。在我给出的第二个php示例中,我只移动了等式的位置,嵌套在三元表达式的if true
部分,而不是if false
部分。我不明白为什么第一种方法如果第二种方法不起作用。这似乎更像是一个错误,而不是"这就是它应该如何工作",而我只是误解了事物,我必须这样做。
注意:我知道我可以用括号括起来强制表达式按照我想要的方式进行评估(显然是正确关联的)。我不是在寻找"如何使这项工作",我想知道为什么不这样做。
答案 0 :(得分:2)
左 - 右 - 关联是将多个运算符串在一起时的优先级,例如A + B + C
。
对于三元运算符A ? B : C
,这只有在额外的三元运算符替换C
部分(或A
部分)时才有意义:
A ? B : X ? Y : Z
(A ? B : X) ? Y : Z <-- left-associative (PHP)
A ? B : (X ? Y : Z) <-- right-associative (Java, C, C++, C#, Perl)
如果在中间插入额外的三元运算符(替换B
),它只能有一个含义:
A ? X ? Y : Z : C
A ? (X ? Y : Z) : C
这就是PHP和Java就第二个问题达成一致的原因。没有应用左/右关联规则。