PHP實現單例模式


單例模式的定義:保證一個類只有一個實例,並提供一個訪問它的全局訪問點。

PHP實現單例模式:

class Singleton
{
    //創建靜態私有的變量保存該類對象
    static private $instance;

    //防止使用new直接創建對象
    private function __construct(){}

    //防止使用clone克隆對象
    private function __clone(){}

    static public function getInstance()
    {
        //判斷$instance是否是Singleton的對象,不是則創建
        if (!self::$instance instanceof self) {
            self::$instance = new self();
        }
        return self::$instance;
    }

    public function test()
    {
        echo "我是一個單例模式";
    }
}

$sing = Singleton::getInstance();
$sing->test();
$sing2 = new Singleton(); //Fatal error: Uncaught Error: Call to private Singleton::__construct() from invalid context in
$sing3 = clone $sing; //Fatal error: Uncaught Error: Call to private Singleton::__clone() from context

 

PHP實現單例模式有什么意義:

沒有意義,PHP沒有線程概念,沒有異步代碼,不會常駐內存,所有的修改自己都可以控制,不需要保證類只有一個實例,有什么意義?你要在數據庫連接類中使用單例模式,萬一有的頁面需要連接兩個數據庫怎么辦?

真的沒有意義嗎?

當然不是。

看下代碼:

class A
{
    public function show()
    {
        echo "我是A類函數,要調用B類函數和C類函數<br/>";
        $b = new B();
        $b->show();
        $c = new C();
        $c->show('A');
    }
}

class B
{
    public function show()
    {
        echo "我是B類函數,要調用C類函數<br/>";
        $c = new C();
        $c->show('B');
    }
}

class C
{
    public function show($parm)
    {
        echo "我是C類函數,現在被{$parm}類函數調用<br/>";
    }
}

$a = new A();
$a->show();

 這里調用的過程中,創建了兩個C類對象,我們知道,new是很耗資源的一種操作,如果這里C類是單例模式,只new一次就行了,這個想法是可以的,但我們一般用另外一種方法來做:

class A
{
    public function show()
    {
        echo "我是A類函數,要調用B類函數和C類函數<br/>";
        $b = Unit::getInstance('B');
        $b->show();
        $c = Unit::getInstance('C');
        $c->show('A');
    }
}

class B
{
    public function show()
    {
        echo "我是B類函數,要調用C類函數<br/>";
        $c = Unit::getInstance('C');
        $c->show('B');
    }
}

class C
{
    public function __construct()
    {
        echo "C類對象被創建<br/>";
    }

    public function show($parm)
    {
        echo "我是C類函數,現在被{$parm}類函數調用<br/>";
    }
}

class Unit
{
    static public function getInstance($class)
    {
        static $arr = null;
        if (!isset($arr[$class]) || !$arr[$class] instanceof $class) {
            $arr[$class] = new $class();
        }
        return $arr[$class];
    }
}

$a = new A();
$a->show();

執行代碼可以看到,C類只被實例化了一次,再實際操作中,我們一般使用靜態變量保存對象,變相實現單例模式。

單例模式和使用靜態變量實現單例的區別:

單例模式強制類只能創建一個對象,一般是安全或者統一界面展示的時候會用到,而靜態變量實現單例主要是為了減少new操作,並不能阻止new和clone操作。

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM