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()。
立即学习“PHP免费学习笔记(深入)”;
-
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,变成“每个子类一个单例”,不是你想要的全局唯一。











