单例模式

什么是单例模式

单例模式,是一种常见的软件设计模式。在应用这个模式时,单例对象的类必须保证只有一个实例存在。

单例模式的优点

  • 减少频繁创建,节省了cpu。
  • 静态对象公用,节省了内存。
  • 功能解耦,代码已维护。

单例模式的应用

实际项目中像数据库查询,日志输出,全局回调,统一校验等模块。这些模块功能单一,但是需要多次访问,如果能够全局唯一,多次复用会大大提升性能。这也是单例存在的必要性。

如何设计单例模式

通过上面的描述,单例的核心是,实例一次生成,全局唯一,多次调用。因此在单例模式必须包含三要素。

  1. 私有化构造函数,私有化clone。也就是不能new,不能clone【唯一】
  2. 拥有一个静态变量,用于保存当前的类。【唯一如何保存】
  3. 提供一个公共的访问入口。【可以访问】

PHP实现

class Singleton
{
    // 私有化构造方法
    private function __construct()
    {

    }
    
    // 私有化clone方法
    private function __clone()
    {
    
    }


    // 保存实例的静态对象
    public static $singleInstance;

    /**
     * 声明静态调用方法
     * 目的:保证该方法的调用全局唯一
     *
     * @return  Singleton
     */
    public static function getInstance()
    {
        if (!self::$singleInstance) {
            self::$singleInstance = new self();
        }

        return self::$singleInstance;
    }


    // 调用单例的方法
    public function singletonFunc()
    {
        echo "call single ton method";
    }

}

$singleInstance =  Singleton::getInstance();
$singleInstance->singletonFunc();

$singleInstance2 =  Singleton::getInstance();
$singleInstance2->singletonFunc();

// 校验是否是一个实例
var_dump($singleInstance === $singleInstance2);  // true ,一个对象

php中static属性和方法的继承问题(补充知识)

class Base
{
    public static $var = 'var';

    public static function testStaticFun()
    {
        echo 'func';
    }
}

class A extends Base
{
    public function testSelf()
    {
        echo self::$var;
    }

    public function testParent()
    {
        echo parent::$var;
    }

    public function setSelf()
    {
        self::$var = 'self';
    }

    public function setParent()
    {
        parent::$var = 'parent';
    }

    public static function testStaticFun()
    {
        parent::testStaticFun();
        echo 'over';
    }
}

$objA = new A();     
  
$objA->testSelf();    // var
$objA->testParent();  // var

$objA->setSelf();
$objA->testSelf();    // self
$objA->testParent();  // self
echo Base::$var;      // self

$objA->setParent();
$objA->testSelf();    // parent
$objA->testParent();  // parent
echo Base::$var;      // parent

Base::testStaticFun();    // func
A::testStaticFun();       // func over

 

  1. 静态成员可以使用 访问控制关键字修饰,可以被继承和重写
    也就就说子类可以继承到父类的静态变量、方法…遵循“关键字”规则
  2. 如果子类没有重写,那么子类调用的实际是父类的静态方法
    这个也很简单明了,继承的基本规则,没什么好解释的
  3. 静态成员持有者是类不是对象,所以类的多个实例共享同一个静态属性的,在一个实例中修改静态属性会影响到另一个实例中的静态属性。
    这个就是重点了,好好理解下,单列模式也是利用这个特性。

所以代码中 self::$var 和 parent::$var 其实指向的都是父类中的$var
最后举个栗子:
单例模式下,$var你可以理解为数据库连接,父类生成一个连接后,继承的子类直接使用这个连接即可连接使用数据库。
但是如果子类自己生成一个连接$var,那么使用的时候就要区分是self::$var还是parent::$var,因为俩个值可能不一样,连接的数据库也不一样,万一是主从呢。

posted @ 2019-08-20 18:00  zswordsman  阅读(226)  评论(0编辑  收藏  举报