我今天在一些PHP代码中看到了这个:

$items = $items ?: $this->_handle->result('next', $this->_result, $this);

我不熟悉这里使用的?:运算符。它看起来像一个三元运算符,但是省略了如果谓词为真则求值的表达式。这是什么意思?


当前回答

如果左操作数为真,则计算为左操作数,否则计算为右操作数。

在伪代码,

foo = bar ?: baz;

粗略地解决

foo = bar ? bar : baz;

or

if (bar) {
    foo = bar;
} else {
    foo = baz;
}

不同的是,这个条只会被求值一次。

你也可以使用它来对foo进行“自检”,如你发布的代码示例所示:

foo = foo ?: bar;

如果foo为null或falsey,则将bar分配给foo,否则将保持foo不变。

更多例子:

<?php
    var_dump(5 ?: 0); // 5
    var_dump(false ?: 0); // 0
    var_dump(null ?: 'foo'); // 'foo'
    var_dump(true ?: 123); // true
    var_dump('rock' ?: 'roll'); // 'rock'
?>

顺便说一下,它叫做猫王运算符。

其他回答

是的,这是PHP 5.3中的新功能。如果测试表达式的值为TRUE,则返回该表达式的值;如果测试表达式的值为FALSE,则返回替代值。

我认为目的是有条件执行:

$a ?: func(); 

只有当$a的值解析为FALSE时,func()中的results才会执行。它可以用作…的缩写

if(!$a){
    func();
}

赋值是可选的$a = $a ?: func()就像:

if(!$a){
    $a = func();
}

另一个重要的考虑:Elvis操作符破坏了Zend Opcache标记化过程。这是我很不容易才发现的!虽然在后来的版本中可能已经修复了这个问题,但我可以确认PHP 5.5.38(内置Zend Opcache v7.0.6-dev)中存在这个问题。

如果你发现你的一些文件在Zend Opcache中“拒绝”缓存,这可能是其中一个原因…希望这能有所帮助!

查看文档:

从PHP 5.3开始,可以省略三元操作符的中间部分。表达式expr1 ?:如果expr1的值为TRUE,则expr3返回expr1,否则返回expr3。

如果左操作数为真,则计算为左操作数,否则计算为右操作数。

在伪代码,

foo = bar ?: baz;

粗略地解决

foo = bar ? bar : baz;

or

if (bar) {
    foo = bar;
} else {
    foo = baz;
}

不同的是,这个条只会被求值一次。

你也可以使用它来对foo进行“自检”,如你发布的代码示例所示:

foo = foo ?: bar;

如果foo为null或falsey,则将bar分配给foo,否则将保持foo不变。

更多例子:

<?php
    var_dump(5 ?: 0); // 5
    var_dump(false ?: 0); // 0
    var_dump(null ?: 'foo'); // 'foo'
    var_dump(true ?: 123); // true
    var_dump('rock' ?: 'roll'); // 'rock'
?>

顺便说一下,它叫做猫王运算符。