单例模式:即只有一个实例。确保某一个类只有一个实例,而且自行实例化并向整个系统提供这个实例。
1.为什么使用单例模式?
(1)php的应用主要在于数据库的应用。一个应用中存在大量的数据库操作。在使用面向对象的开发时,如果使用单例模式,则可以避免大量的new操作消耗的资源,还可以减少数据库连接这样就不会出现too many connections情况
(2)如果系统中需要一个类来全局控制某些配置信息,可以使用单例模式实现。(或者一些配置信息放入配置文件中yii框架)
(3)便于调试。所有的数据库操作放入这个类中,在这一个类中输出日志或者打印。
2.实现方法?
简称 三私一公
(1)私有化一个属性存放唯一的实例
(2)私有化构造方法,这样在类外不能使用new关键字实例化这个类(因为private)
(3)私有化克隆方法
(4)公有化静态方法,用于向系统提供这个实例
3.代码实现
(1)
class Singleton{
//存放实例
private static $_instance = null;
//私有化构造方法、
private function __construct(){
echo "单例模式的实例被构造了";
}
//私有化克隆方法
private function __clone(){
}
//公有化获取实例方法
public static function getInstance(){
if (!(self::$_instance instanceof Singleton)){ //instanceof判断这个对象是否属于这个类,返回true或false
self::$_instance = new Singleton(); //在本类中可以new,private可以在本类中进行调用
}
return self::$_instance;
}
public function select(){
}
}
$singleton=Singleton::getInstance();
$singleton->select(); //对象调用方法
(2)php引进了trait,使用trait实现一下
a.这个类用trait声明,而不是class声明。trait声明的类不能实例化,也就是不能使用new去创建
b.那么想要使用这个trait类的其他类中,使用use关键字引入即可
c.trait、基类(正常继承的父类class)、本类中同名的属性和方法,本类覆盖trait类,trait类覆盖基类
d.组合多个trait,则use trait1,trait2即可
e.不同的trait中,有同名的属性或者方法冲突,可以使用instanceof或者as解决,instanceof是进行代替,as是取别名
class dog{
use trait1,trait2{
trait1::eat insteadof trait2;
trait1::drive insteadof trait2;
trait2::eat as eaten;
trait2::drive as driven;
}
//如果没有只有一个trait或者没有同名冲突的话,use即可
use trait1,trait2;
}
下面用trait具体实现单例
Trait Singleton{
//存放实例
private static $_instance = null;
//私有化克隆方法
private function __clone(){
}
//公有化获取实例方法
public static function getInstance(){
$class = __CLASS__;
if (!(self::$_instance instanceof $class)){
self::$_instance = new $class();
}
return self::$_instance;
}
}
class DB {
private function __construct(){
echo __CLASS__.PHP_EOL;
}
}
class DBhandle extends DB {
use Singleton; //use引入
private function __construct(){
echo "单例模式的实例被构造了";
}
public function select(){
}
}
$handle=DBhandle::getInstance();
$handle->select();
单例模式可以被破坏吗?
1.反射机制可以破坏单例模式。php5和php7中引入了反射类,使用这个反射类,将其他类以参数的方式传入,就可以或者传入的类里面的成员和方法,包括private修饰符。这样就能调用构造方法,数据库单例模式中构造方法里面为数据库连接操作,是一些初始化的一些东西,这样反射了这个类,就可以调用了构造方法了,就等于又有了一个实例,这样就破坏了单例的设计思想。
2.克隆对象也可以,使用关键子clone克隆对象。克隆出来的对象与原对象没有任何关系,它是把原来的对象从当前的位置重新复制了一份,也就是相当于在内存中新开辟了一块空间,那就有两个对象了即两个实例破坏了单例。$objClone = clone $obj。所以单例类中要定义__clone()防止外面进行克隆。