PHP单例模式必须用private static $instance,因其确保仅类内可读写,防止外部篡改破坏全局唯一性;构造、克隆、反序列化方法均需private,getInstance()须public static且用self::保证父类单例契约。
PHP 中用 :: 实现单例模式,本质是靠静态属性 + 静态方法 + 作用域操作符控制类的实例化入口,不是语法糖,而是明确切断 new 的公开路径。
private static $instance 而非 public
单例的核心约束是“全局唯一实例”,如果把 $instance 设为 public,外部就能随意赋值或清空,比如 MyClass::$instance = null; 或 MyClass::$instance = new MyClass();,直接破坏单例语义。静态属性必须配合 private 才能真正封装。
private static $instance 确保只有本类内部可读写protected 允许子类访问,但子类可能绕过构造逻辑,不推荐用于基础单例const,因为实例需在运行时创建,而常量必须是编译期确定值getInstance() 必须是 public static 方法这是外界唯一合法获取实例的门面(Facade)。它负责检查、创建、返回——所有逻辑收束于此。若设为 private 或 protected,外部根本调用不到;若非 static,则需先有实例才能调用,陷入循环依赖。
getInstance() 内部用 self::$instance === null 判断是否已存在new self()(而非 new static()),否则在继承场景下可能返回子类实例,破坏父类单例契约private,则连 new self() 都无法在外部调用——这正是你想要的private 或 protected
这是防止绕过 getInstance() 的最后一道防线。只要构造方法不是 public,任何 new MyClass() 都会触发 Fatal error: Uncaught Error: Call to private MyClass::__construct()。
private __construct():最严格,连子类都无法继承或调用protected __construct():允许子类扩展,但子类也必须自己实现单例逻辑,否则无法复用父类 getInstance()
clone 和 __wakeup 也设为 private,防止反序列化或克隆破环单例class DatabaseConnection
{
private static $instance = null;
private function __construct() {}
private function __clone() {}
private function __wakeup() {}
public static function getInstance()
{
if (self::$instance === null) {
self::$instance = new self();
}
return self::$instance;
}
}
注意:self:: 和 static:: 在单例中行为不同。用 self:: 才能确保始终操作当前类的静态属性;若父类用了 static::,子类调用时会写入子类自己的 $instance,变成“每个子类一个单例”,不是你想要的全局唯一。