17370845950

如何用::实现单例模式_php静态方法与作用域操作符应用【技巧】
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)。它负责检查、创建、返回——所有逻辑收束于此。若设为 privateprotected,外部根本调用不到;若非 static,则需先有实例才能调用,陷入循环依赖。

  • 典型实现中,getInstance() 内部用 self::$instance === null 判断是否已存在
  • 创建时必须用 new self()(而非 new static()),否则在继承场景下可能返回子类实例,破坏父类单例契约
  • 若构造方法是 private,则连 new self() 都无法在外部调用——这正是你想要的

构造方法必须声明为 privateprotected

这是防止绕过 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,变成“每个子类一个单例”,不是你想要的全局唯一。