教你用两行代码给PHP7添加一个“非空合并”语法糖

教你用两行代码给PHP7添加一个“非空合并”语法糖,第1张

教你用两行代码给PHP7添加一个“非空合并”语法糖 我们知道从 PHP 5.3 起三元运算符 ? : 有一个写法简洁写法是这样的:

<?php
$a = 0;
$b = $a ?: 1; # $b === 1

这实际上相当于:

<?php
$a = 0;
$b = $a ? $a : 1; # $b === 1

在 PHP5 中,语法分析是这样写的:

|   expr '?' { zend_do_begin_qm_op(&, & TSRMLS_CC); }
    expr ':' { zend_do_qm_true(&, &, & TSRMLS_CC); }
    expr     { zend_do_qm_false(&$$, &, &, & TSRMLS_CC); }
|   expr '?' ':' { zend_do_jmp_set(&, &, & TSRMLS_CC); }
    expr     { zend_do_jmp_set_else(&$$, &, &, & TSRMLS_CC); }

在 PHP7 中,由于 AST(抽象语法树)的引入,语法分析有些简化:

|   expr '?' expr ':' expr
        { $$ = zend_ast_create(ZEND_AST_CONDITIONAL, , , ); }
|   expr '?' ':' expr
        { $$ = zend_ast_create(ZEND_AST_CONDITIONAL, , NULL, ); }

PHP7 中语法分析之后都是写到 AST 的节点上。从上面可以看出,简化的 ?: 和完整的三元表达式的区别就是节点中间的值为 NULL。

PHP7 添加了一个合并 *** 作符(T_COALESCE),用于简化 isset 的条件判断:

<?php
$b = $a ?? 1;

它相当于:

<?php
$b = isset($a) ? $a : 1;

仅仅是 isset 判断,在 为等值时还是会返回b 的值还是为 0

这个 *** 作符的语法分析语句是:

|   expr T_COALESCE expr
        { $$ = zend_ast_create(ZEND_AST_COALESCE, , ); }

如果想将 isset 换成 empty 的效果,也就是说在变量不存在或转换成 boolean 后为 false 都赋予其他值,需要这样写:

<?php
$b = $a ?? 1 ?: 1;

显然上面的表达式中中间一部分稍微有些多余,那么做些简化呢?

现在我想添加一个语法 ??: ,它的作用是对变量做 empty 的判断。也就是说达到上面 $a ?? 1 ?: 1 的效果:

<?php
$b = $a ??: 1;

改起来很简单,只需要将 ?: ?? 的分析合并一下(注意这里和上面所有的地方 等符号的数字表示的都是变量或者常量出现的位置顺序):

|   expr T_COALESCE ':' expr
        { $$ = zend_ast_create(ZEND_AST_CONDITIONAL,
            zend_ast_create(ZEND_AST_COALESCE, , ), NULL, ); }

仅仅只有两句,因为并没有在词法分析器中添加 Token,所以只能算是个语法糖。

重新编译一下之后就能看到效果啦。

测试:

$ /usr/local/php/bin/php -r "$a = 0; echo $a ?? 1, PHP_EOL;"
0
$ /usr/local/php/bin/php -r "$a = 0; echo $a ??: 1, PHP_EOL;"
1

以上就是教你用两行代码给PHP7添加一个“非空合并”语法糖的详细内容,

欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/langs/679741.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2022-04-20
下一篇 2022-04-20

发表评论

登录后才能评论

评论列表(0条)

保存