$123abc 是非法的,因php变量名首字符必须为字母或下划线,数字开头会在词法分析阶段报错;合法变量名需满足:首字符为字母或下划线、后续可含字母数字下划线、非保留字。

PHP变量名不能以数字开头
直接说结论:$123abc 是非法的,PHP 解析器会报 Parse error: syntax error, unexpected '123abc' (T_STRING)。这不是运行时错误,而是在词法分析阶段就拒绝解析——根本不会进入执行流程。
合法变量名的三个硬性条件
PHP 变量名必须同时满足:
- 第一个字符只能是字母(a–z、A–Z)或下划线
_ - 后续字符可以是字母、数字(0–9)或下划线
_ - 不能是 PHP 保留字,比如
$class、$function虽然语法上合法,但容易引发混淆或未来兼容风险
所以 $abc123、$_count、$user_id 都行;$123、$99bottles、$-price 全部非法。
为什么设计成这样?和其它语言对比
这其实是绝大多数 C 系语言(C、Java、JavaScript、Python)共用的标识符规则,核心是为了让词法分析器能无歧义地切分 token。比如解析 3x 时,如果不禁止数字开头,解析器无法判断这是变量名还是数字字面量 123 后跟一个未定义变量 x。
立即学习“PHP免费学习笔记(深入)”;
注意:这和常量不同——define('123ABC', 'ok') 是允许的,因为常量名是字符串参数,不走变量名校验逻辑;但 const 123ABC = 'ok' 同样非法,因为类常量也遵守标识符规则。
实际开发中容易踩的坑
常见于动态构造变量名场景,比如从数据库字段、URL 参数、JSON 键名生成变量时没做清洗:
- 从 JSON 解析出键为
"2024_total",直接写$$key会崩,得先转成$var_2024_total或用数组代替 - 用
extract($_GET)时,如果 URL 带?123=abc,PHP 会静默跳过这个键,不报错也不赋值——这点非常隐蔽,调试时容易漏掉 - IDE 或静态分析工具(如 PHPStan)通常不会警告变量名非法,因为它们依赖已解析 AST;错误只在运行前暴露,属于“写完就炸”型问题
最稳妥的做法:别依赖动态变量名,优先用关联数组或对象属性。真要动态,加一层校验:preg_replace('/^[^a-zA-Z_]/', '_', $key),至少兜住开头。











