
`??` 是 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免费学习笔记(深入)”;
<?php // 从超全局变量安全取值 $user = $_GET['user'] ?? 'anonymous'; // 若 $_GET['user'] 不存在或为 null,则用 'anonymous' // 变量自身回退(常用于函数参数或配置兜底) $theme = $config['theme'] ?? 'light'; // 支持链式调用(PHP 7.4+ 更支持多级,但 PHP 7.0+ 已支持嵌套 ??) $city = $_GET['location']['city'] ?? $_GET['city'] ?? 'Beijing'; // 注意:空字符串、0、false 不会被替换! $score = 0; echo $score ?? 100; // 输出 0(不是 100),因为 0 !== null echo '' ?? 'default'; // 输出空字符串(不是 'default') ?>
⚠️ 重要注意事项:
- ?? 不等价于 empty():empty($x) 会将 0、'0'、[]、false、null、'' 均视为“空”,而 ?? 仅对 null 和“未定义”生效;
- ?? 也不完全等价于 isset():isset($x) 仅检查变量是否存在且非 null,但 ?? 是表达式运算符,可直接参与赋值与链式计算;
- 若左侧表达式引发警告(如访问未定义数组键),?? 会静默抑制该警告(这是其设计特性,但需确保逻辑合理);
- 不能用于函数调用左侧(如 (foo()) ?? 'bar' 合法,但 foo() ?? 'bar' 中若 foo() 未定义会报致命错误)。
? 进阶技巧:结合 ??=(空合并赋值运算符,PHP 7.4+)实现懒初始化:
<?php $data ??= []; // 等价于 $data = $data ?? []; $cache['user'] ??= getUserFromDB(); ?>
总之,?? 是 PHP 7 带来的实用语法糖,显著提升代码可读性与健壮性。掌握其与 isset()、empty() 的本质区别,能帮你写出更精准、更现代的 PHP 代码。










