
`??` 是 php 7 引入的空合并运算符(NULL coalescing operator),用于安全获取变量值——当左侧操作数存在且不为 `null` 时返回其值,否则返回右侧默认值,等价于 `isset($a) && $a !== null ? $a : $b`,但更简洁、可链式使用。
在 php 开发中,频繁判断变量是否已定义且非 null 是常见需求,传统写法往往冗长且易出错。例如,从 $_GET 或数组中取值时,需层层嵌套 isset() 和三元运算符:
// ❌ 传统写法(繁琐且易漏判) $username = isset($_GET['user']) ? $_GET['user'] : 'guest'; // ❌ 更复杂场景(如多级数组) $country = isset($_GET['profile']) && isset($_GET['profile']['address']) ? $_GET['profile']['address']['country'] : 'unknown';
而空合并运算符 ?? 正是为此优化而生。它的语法简洁明了:
$result = $expr1 ?? $expr2;
语义为:若 $expr1 已声明(exists)且不为 null,则返回 $expr1 的值;否则返回 $expr2 的值。注意:它仅检查 null 和“未定义”,不触发 empty() 判断(即 0、”、false 等“falsy”值仍会被原样返回)。
✅ 正确用法示例:
立即学习“PHP免费学习笔记(深入)”;
⚠️ 重要注意事项:
- ?? 不等价于 empty():empty($x) 会将 0、’0’、[]、false、null、” 均视为“空”,而 ?? 仅对 null 和“未定义”生效;
- ?? 也不完全等价于 isset():isset($x) 仅检查变量是否存在且非 null,但 ?? 是表达式运算符,可直接参与赋值与链式计算;
- 若左侧表达式引发警告(如访问未定义数组键),?? 会静默抑制该警告(这是其设计特性,但需确保逻辑合理);
- 不能用于函数调用左侧(如 (foo()) ?? ‘bar’ 合法,但 foo() ?? ‘bar’ 中若 foo() 未定义会报致命错误)。
? 进阶技巧:结合 ??=(空合并赋值运算符,PHP 7.4+)实现懒初始化:
总之,?? 是 PHP 7 带来的实用语法糖,显著提升代码可读性与健壮性。掌握其与 isset()、empty() 的本质区别,能帮你写出更精准、更现代的 PHP 代码。