php变量读取比常量快15%~25%,因变量直接从符号表取值,而常量需哈希查找;类常量略快于全局常量,但仍慢于局部变量;define()与const性能无差别但行为不同。

PHP变量读取比常量快还是慢?
常量在PHP中不是“编译期展开”,而是运行时查表——defined() 和 constant() 都要走哈希查找,而变量直接从符号表(symbol table)里取值,路径更短。实测在PHP 8.0+环境下,纯读取场景下,变量访问比常量快约15%~25%,尤其在循环内差异明显。
但这个差距只在高频、简单读取时可测;一旦涉及函数调用、数组访问或对象属性,这点差异立刻被淹没。
- 常量名解析需额外调用
zend_get_constant,变量则走zend_fetch_var,底层路径更短 -
define('FOO', 'bar')创建的是“运行时常量”,不是宏替换,不会被ZEND优化掉 - 类常量(
self::FOO)比全局常量(FOO)略快,因作用域更明确,但依然慢于局部变量
define() vs const 关键区别在哪?
两者性能几乎无差别(都是注册到常量表),但行为和适用场景完全不同,选错会直接报错或逻辑异常。
-
define()是函数,支持动态名称和表达式:define('CACHE_' . $env, true)✅;const CACHE_$env = true❌(语法错误) -
const在类外只能用于文件顶层,不能放在if或函数内;define()可以 - 类中必须用
const(public const FOO = 1;),define()对类常量无效 - PHP 7.4+ 支持
const定义数组,define()不支持(会转成字符串)
为什么 var_dump(FOO) 有时输出 int 而有时是 string?
这不是类型问题,是常量未定义时的静默降级:当 FOO 未定义,PHP会把它当作字符串字面量 'FOO' 处理(仅在非严格模式下),var_dump() 就真输出字符串。这容易掩盖拼写错误。
立即学习“PHP免费学习笔记(深入)”;
- 开启
error_reporting(E_ALL)后,未定义常量会触发E_NOTICE: Use of undefined constant FOO - 用
defined('FOO')显式检查,比直接使用更安全 - 现代写法推荐优先用
match或配置数组替代大量全局常量,避免命名污染和误读
常量真的不可变吗?
语言层面“不可变”仅指不能用 = 重新赋值,但有三个常见绕过方式,导致实际行为不符合预期:
- 用
define()重复定义同名常量(PHP 8.0+ 报Fatal error,但 PHP 7.x 默认静默忽略,后一次无效) - 扩展如
apcu或opcache的预加载(opcache.preload)可能把常量固化,导致define()失效 - 反射可修改类常量(
ReflectionClass::getConstants()只读,但通过zend_alter_ini_entry等底层手段在扩展中可破环——不建议,且不稳定)
真正需要“硬不可变”的场景(如安全密钥),应该用 readonly 属性或环境变量 + getenv(),而不是依赖常量语义。











