单例模式,正如其名,允许我们创建一个而且只能创建一个对象的类。
这在整个系统的协同工作中非常有用,特别明确了只需一个类对象的时候。
那么,为什么要实现这么奇怪的类,只实例化一次?
在很多场景下会用到,如:配置类、Session类、Database类、Cache类、File类等等。
这些只需要实例化一次,就可以在应用全局中使用。
本文我们以数据库类为例。
class Database
{
// 声明$instance为私有静态类型,用于保存当前类实例化后的对象
private static $instance = null;
// 数据库连接句柄
private $db = null;
// 构造方法声明为私有方法,禁止外部程序使用new实例化,只能在内部new
private function __construct($config = array())
{
$dsn = sprintf('mysql:host=%s;dbname=%s', $config['db_host'], $config['db_name']);
$this->db = new PDO($dsn, $config['db_user'], $config['db_pass']);
}
// 这是获取当前类对象的唯一方式
public static function getInstance($config = array())
{
// 检查对象是否已经存在,不存在则实例化后保存到$instance属性
if(self::$instance == null) {
self::$instance = new self($config);
}
return self::$instance;
}
// 获取数据库句柄方法
public function db()
{
return $this->db;
}
// 声明成私有方法,禁止克隆对象
private function __clone(){}
// 声明成私有方法,禁止重建对象
private function __wakeup(){}
}
再通过getInstance()
方法使用类对象,
$config = array(
'db_name' => 'test',
'db_host' => 'localhost',
'db_user' => 'root',
'db_pass' => 'root'
);
$db1 = Database::getInstance($config);
var_dump($db1);
$db2 = Database::getInstance($config);
var_dump($db2);
$db3 = Database::getInstance($config);
var_dump($db3);
单例模式中,不同对象获得的资源ID是一样的。
也就是说,虽然我们用getInstance()
获取Database
类对象3次,其实引用的是一个内存空间,PDO也只连接了数据库一次。
特点
单例模式的特点是 4私1公 : 一个私有静态属性,构造方法私有,克隆方法私有,重建方法私有,一个公共静态方法。其他方法根据需要增加。
最基础的单例模式代码如下:
class Singleton
{
private static $instance = null;
public static function getInstance()
{
if(self::$instance == null) {
self::$instance = new self();
}
return self::$instance;
}
private function __construct(){}
private function __clone(){}
private function __wakeup(){}
}
$instance
用以保存类的实例化,getInstance()
方法提供给外部本类的实例化对象: