每一种设计模式都是为了解决特定的问题,单例模式从名字就可以看出,是软件系统中只需要一个对象时使用。
如果一个类在系统中只能有一个实例,可以通过如下代码实现
<?php
/**
* Class Singleton
* @datetime 2020/7/12 10:08 PM
* @author roach
* @email jhq0113@163.com
*/
class Singleton
{
/**
* @var Singleton
* @datetime 2020/7/12 10:09 PM
* @author roach
* @email jhq0113@163.com
*/
private static $_instance;
/**
* Singleton constructor.
*/
private function __construct(){}
/**
* @datetime 2020/7/12 10:08 PM
* @author roach
* @email jhq0113@163.com
*/
private function __clone(){}
/**
* @return Singleton
* @datetime 2020/7/12 10:08 PM
* @author roach
* @email jhq0113@163.com
*/
public static function getInstance()
{
if(is_null(self::$_instance)) {
self::$_instance = new self();
}
return self::$_instance;
}
}
以上代码是一个标准单例模式的实现,其中的关键实现有一下三点
但是在实际的开发中,我们一般不会用以上代码来实现单例,原因有以下两点
new self()
,如果其他类也想实现单例必须复制以上代码上面第一个问题,我们可以通过静态延时绑定来解决,创建对象时,我们将
new self()
改为new static()
,这样如果有其他类也想实现单例,直接继承该类就好,但是第二个问题还是无法解决那么第二个问题怎么解决呢?
在实际项目开发中一般使用依赖注入方式实现,单独提供一个DI容器,靠配置注入方式实现,这样就不是直接依赖关系了。下面提供一个简单的DI容器类
<?php
/**
* Class Di
* @datetime 2020/7/12 10:29 PM
* @author roach
* @email jhq0113@163.com
*/
class Di
{
/**
* @var array
* @datetime 2020/7/12 10:23 PM
* @author roach
* @email jhq0113@163.com
*/
private static $_pool = [];
/**
* @param string $key
* @param array $config
* @datetime 2020/7/12 10:24 PM
* @author roach
* @email jhq0113@163.com
*/
public static function set($key, $config = [])
{
self::$_pool[ $key ] = $config;
}
/**
* @param string $key
* @return mixed|null
* @throws Exception
* @datetime 2020/7/12 10:29 PM
* @author roach
* @email jhq0113@163.com
*/
public static function get($key)
{
if(!isset(self::$_pool[ $key ])) {
return null;
}
$config = self::$_pool[ $key ];
if(isset($config['class'])) {
$class = $config['class'];
unset($config['class']);
if(!class_exists($class)) {
throw new \Exception($class.'不存在');
}
$object = new $class();
foreach ($config as $property => $value) {
$object->$property = $value;
}
self::$_pool[ $key ] = $object;
}
return self::$_pool[ $key ];
}
}
下面是使用容器创建一个
User
类,并且注入了nickname
属性
<?php
/**
* Class User
* @datetime 2020/7/12 10:30 PM
* @author roach
* @email jhq0113@163.com
*/
class User
{
/**
* @var string
* @datetime 2020/7/12 10:30 PM
* @author roach
* @email jhq0113@163.com
*/
public $nickname;
}
Di::set('user', [
'class' => User::class,
'nickname' => uniqid()
]);
echo Di::get('user')->nickname.PHP_EOL;
通过以上容器实现单例,解决了上面提到的第二个问题,我们每个类如果想在项目中实现单例,不用特殊的实现,也不用继承某个类,我们在调用端通过容器的
get
方法得到对象,这里的依赖关系属于依赖注入,get
方法里的对象是通过配置注入进来的,并且全局是一个单例模式。作者开源了一个功能更丰富的容器,封装在了
jhq0113/roach
中,我们可以通过composer
方式安装,代码开源地址如下
https://github.com/jhq0113/roach
jhq0113/roach
使用简单,代码精简,整个代码库纯代码大小为60K,composer
安装方式
composer require jhq0113/roach