【PHP设计模式 05_DanLi.php】单例模式

<?php 
/**
 * 【单例模式】
 * 总结:防止外部new对象;防止子类继承;防止克隆。
 */

header("Content-type: text/html; charset=utf-8");

/******* NO.1 普通的可以调用的类 ******/
//先定义一个类,实例化两次,看看是否全等(===)
class sigleA{
    
}
$sa1 = new sigleA();
$sa2 = new sigleA();
//注意:当两个对象是一个的时候,才会全等
//此处输出 “sa-不是一个对象”
if($sa1 === $sa2){
    echo 'sa-是一个对象<br>';
}else{
    echo 'sa-不是一个对象<br>';
}

/********************* NO.2 封锁new操作 ********************/
//getIns 预先判断实例
class sigleB{
    protected static $ins = null;
    public static function getIns(){ //getIns 获取实例
        if(self::$ins === null){
            //把【自身对象】 赋给 一个自己的 【静态属性】
            self::$ins = new self();
        }
        return self::$ins;
    } 
    //保护的 构造函数
    protected function __construct(){
        
    }
}
//再去判断两个对象
$sb1 = sigleB::getIns();
$sb2 = sigleB::getIns();
//此处输出 “sb-是一个对象”
if($sb1 === $sb2){
    echo 'sb-是一个对象<br>';
}else{
    echo 'sb-不是一个对象<br>';
}

////----- 但是,下面用一个新的类继承自上面的 sigleB ...
class multi extends sigleB{
    public function __construct(){
        
    }
}
$m1 = new multi();
$m2 = new multi();
//此处输出 “m-不是一个对象”,原因在于被继承后,__construct 被公开了
if($m1 === $m2){
    echo 'm-是一个对象<br>';
}else{
    echo 'm-不是一个对象<br>';
}


/********************* NO.3 final防止继承 ********************/
class sigleC{
    protected static $ins = null;
    public static function getIns(){
        if(self::$ins === null){
            self::$ins = new self();
        }
        return self::$ins;
    }
    //若方法前加上final,则方法不能被覆盖;
    //若类前加上final,则类不能被继承
    final protected function __construct(){

    }
    //同样的,防止克隆(clone)
    final protected function __clone(){
        
    }
}
$sc1 = sigleC::getIns();
$sc2 = clone $sc1; 
//被克隆了,又产生了多个对象;
//如果没加上 防止克隆的代码,此处输出 “sc-不是一个对象”

//上面加上了 final protected function __clone(),这里就会出错:
//Fatal error: Call to protected sigleC::__clone() from context '' in ...
if($sc1 === $sc2){
    echo 'sc-是一个对象<br>';
}else{
    echo 'sc-不是一个对象<br>';
}
原文地址:https://www.cnblogs.com/rxbook/p/6002976.html