设计模式——单例模式

单例模式:作为对象的创建模式,它提供了一种创建对象的最佳方式。单例模式确保某一个类只有一个实例,而且自行实例化并向整个系统全局地提供这个实例。它不会创建实例副本,而是会向单例类内部存储的实例返回一个引用。

主要应用场合:线程池、缓存、日志对象、对话框、打印机、数据库操作、显卡的驱动程序常被设计成单例。使用单例模式可以避免大量的new操作。因为每一次new操作都会消耗系统和内存的资源。

例如下面代码:


class UserModel
{
    public function __construct ()
    {
    }

    public function getUser()
    {
        $userInfo = [];
        return $userInfo;
    }
}

//项目中第一次用用UserModel;
$userInfo1 = new UserModel();

//项目中第二次用用UserModel;
$userInfo2 = new UserModel();

if ($userInfo1 === $userInfo2) {
    echo '实例相同';
} else {
    echo '实例不相同';
}

运行结果:

图片

两次实例化UserModel,会开辟两个内存造成资源浪费。单例就让多次使用UserModel,只用一个内存。

单例模式有以下3个特点:

1.只能有一个实例。

2.必须自行创建这个实例。

3.必须给其他对象提供这一实例。

为符合上面的特点:

1、需要一个保存类的唯一实例的静态成员变量;

2、构造函数和克隆函数私有化,防止外部实例化和克隆;

3、必须提供一个访问这个实例的公共的静态方法(通常为getInstance方法),从而返回唯一实例的一个引用 

通过代码来实现:

class UserModel
{
    //1、定义保存类的唯一实例的静态成员变量
    private static $_instance = null;

    //2、构造函数和克隆函数私有化,防止外部实例化和克隆;
    private function __construct ()
    {

    }

    private function __clone ()
    {

    }

    //3、必须提供一个访问这个实例的公共的静态方法(通常为getInstance方法),从而返回唯一实例的一个引用
    public static function getInstance ()
    {
        if(!(self::$_instance instanceof self)) {
            self::$_instance = new self();
        }
        return self::$_instance;
    }

    public function getUser()
    {
        $userInfo = [];
        return $userInfo;
    }
}

//项目中第一次用用UserModel;
$userInfo1 = UserModel::getInstance();

//项目中第二次用用UserModel;
$userInfo2 = UserModel::getInstance();

if ($userInfo1 === $userInfo2) {
    echo '实例相同';
} else {
    echo '实例不相同';
}

运行结果:

图片

 

从运行结果可以知这种两次使用,并没有重复实例化。看似一个单例已经实现。可这时我们项目中有一个学生用户信息继承了UserModel


class StudentModel extends UserModel
{
    public function __construct ()
    {

    }
    public function __clone ()
    {

    }
}

$userInfo1= new StudentModel();
$userInfo2 = new StudentModel();
$userInfo3 = clone $userInfo2;

if ($userInfo1 === $userInfo2) {
    echo '实例相同<br/>';
} else {
    echo '实例不相同<br/>';
}

if ($userInfo3 === $userInfo2) {
    echo '实例相同<br/>';
} else {
    echo '实例不相同<br/>';
}

运行结果:

图片

现在问题又来了,我们辛辛苦苦创建的单例模型被继承后又失效了。就好比地主老财辛辛苦苦赚的钱谁都不借,结果出了个败家子把钱都花光了。为了让其子类以保持单例的特性。我们应将__constract和__call加上关键字final

class UserModel
{
    //1、定义保存类的唯一实例的静态成员变量
    private static $_instance = null;

    //2、构造函数和克隆函数私有化,防止外部实例化和克隆;
    private final function __construct ()
    {

    }

    private  final function __clone ()
    {

    }

    //3、必须提供一个访问这个实例的公共的静态方法(通常为getInstance方法),从而返回唯一实例的一个引用
    public static function getInstance ()
    {
        if(!(self::$_instance instanceof self)) {
            self::$_instance = new self();
        }
        return self::$_instance;
    }

    public function getUser()
    {
        $userInfo = [];
        return $userInfo;
    }
}

以保证__constract和__call不被覆盖或重写。所以我个人认为一个完整的单例模型应该如上面所写的类。

图片

 

关注快乐程序员公众号,每日分享一点小知识。爱编程,爱生活!

 

 

 

 

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值