静态属性属于类本身、被所有实例共享,首次加载时初始化且不自动重置;静态变量在函数内仅初始化一次、跨调用保留值;二者均不可序列化,且受运行环境(cli/fpm)影响显著。

静态属性在类实例间共享,赋值后会变
PHP 的 static 属性不属于某个对象实例,而是属于类本身。只要程序运行期间未被显式修改或脚本结束,它的值就一直存在且所有实例共用。
常见错误现象:new 多个对象后对静态属性赋值,误以为只影响当前实例;或者在 CLI 模式下反复执行脚本时,发现值“没重置”,其实是每次执行都是新进程,静态属性本来就是从头初始化的——这点常被当成“值变了”,其实是预期行为。
- 静态属性在类首次被加载(autoload 或直接使用)时初始化一次,之后除非代码主动改写,否则不会自动重置
- 可在类内部、子类、外部通过
self::$prop、static::$prop、ClassName::$prop读写 - 若声明时用了表达式(如
static $count = time();),PHP 会报错:静态属性只能用字面量或常量初始化
静态变量在函数内仅初始化一次,后续调用保留上次值
函数内的 static 变量不是全局变量,但生命周期跨越多次函数调用。它在第一次执行到该行时初始化,之后每次进入函数都跳过初始化,直接使用上一次退出时的值。
使用场景:计数器、缓存单次计算结果、避免重复初始化资源(如数据库连接句柄)。
立即学习“PHP免费学习笔记(深入)”;
- 不能用变量、函数调用等动态表达式初始化,例如
static $cache = get_default();是非法的 - CLI 下连续调用同一函数,值持续累加;FPM/Apache 模式下每个请求是独立进程,静态变量每次请求都重新初始化
- 注意与
global区分:静态变量作用域仍限于函数内,外部不可见
静态属性和静态变量都被序列化忽略
调用 serialize() 序列化对象时,静态属性不会被保存;同理,unserialize() 还原对象后,静态属性恢复为类定义时的初始值(或未初始化状态)。
静态变量更彻底:它根本不在对象结构里,序列化完全不感知它的存在。
- 如果依赖静态属性存储关键状态(如登录令牌计数),别指望序列化/反序列化能维持它
- 想持久化这类数据,必须手动提取并存入可序列化的字段,或改用外部存储(Redis、文件等)
-
__sleep()和__wakeup()对静态属性无效,它们只处理普通属性
继承中 static:: 和 self:: 对静态属性的行为差异
子类继承父类静态属性时,如果不重定义,会共享同一份存储;但一旦子类声明了同名静态属性,就变成独立副本。这时用 self:: 总是指向当前类定义的作用域,而 static:: 遵循后期静态绑定(LSB),指向实际调用的类。
容易踩的坑:以为 self::$data 在子类方法里会访问子类自己的静态属性,结果还是改到了父类的。
- 用
static::$data才能实现“谁调用,改谁的”逻辑;用self::$data则永远固定在定义该方法的类上 - PHP 5.3+ 支持 LSB,低版本只能靠传参或抽象方法绕过
- 静态属性不支持
private继承可见性控制:子类无法访问父类private static属性,哪怕在父类方法里用self::访问也不行









