这个报错("Only variables should be passed by reference")本质上源于PHP 5.4及以上版本的一项严格规定:不允许将函数返回值直接传递给需要引用参数的数组函数,例如end、reset等。它属于E_STRICT级别的警告。唯一的解决方案是修改代码逻辑,切勿试图通过调整error_reporting来掩盖问题。

需要明确的是——这个问题的根源并不在XAMPP。触发该警告的原因是PHP 5.4+对函数返回值引用传递的限制被激活。因此,绝对不要试图通过降低error_reporting来隐瞒错误,必须从根本上修改代码。
为什么error_reporting(E_ALL & ~E_NOTICE)不起作用
为什么这种方法无效呢?因为这个错误属于E_STRICT级别,自PHP 5.4起已被纳入E_ALL。默认情况下E_STRICT并不启用,但一旦开发者在配置中显式启用了该级别——例如框架中写了error_reporting(E_ALL | E_STRICT)——那么警告就会显现。仅靠~E_NOTICE无法屏蔽E_STRICT。
error_reporting(E_ALL & ~E_NOTICE)仍会输出E_STRICT警告- 正确的写法应为
error_reporting(E_ALL & ~E_NOTICE & ~E_STRICT)(但不推荐长期使用) - 更常见的情况是:某些CMS(如旧版Drupal、Joomla)或自定义函数中使用了类似
end(explode('/', $path))的写法,PHP不允许将函数返回值直接传递给需要引用参数的函数
end()、reset()、current()等数组指针函数不能直接套用函数返回值
这是最常见的触发场景:试图对一个临时数组结果直接调用指针操作函数。然而这些函数要求必须传入一个变量,而不能是表达式计算出的结果。简单来说,就是需要提供一个“容器”,而非一个“临时值”。
- ❌ 错误写法:
$ext = end(explode('.', $filename)); - ✅ 正确写法:
$parts = explode('.', $filename); $ext = end($parts); - 其他函数同理:
reset()、prev()、next()、key()都有同样限制 - 注意——从PHP 7.4+开始,这类写法会直接报
Fatal error,而不仅仅是警告
如何快速定位出问题的代码行
怎么找到它呢?报错信息通常只提供文件名和行号,不会明确指出具体的嵌套调用。关键在于检查报错堆栈中最后一行是否属于你自己的.php文件,然后查看该行是否用到了上文提到的数组指针函数。
- 打开XAMPP控制面板 → Apache → Config → PHP(php.ini)→ 搜索
error_reporting,确认当前值(通常为E_ALL) - 在报错页面右键查看源码,找到
Warning: Only variables should be passed by reference附近的具体行号 - 如果使用IDE,可以在项目中全局搜索
end(、reset(、current(,逐一检查括号内是函数调用还是变量 - 临时添加一句
ini_set('error_reporting', E_ALL & ~E_STRICT);可以验证是否为该级别导致——但这仅用于排查,绝不能长期保留
真正需要修改的是代码中那些将函数返回值当作变量使用的写法。依赖调整error_reporting来隐藏问题,无异于掩耳盗铃。一旦程序部署到更高版本的PHP环境,随时可能崩溃。
