设计模式 - 关于PHP单例模式,有一点不明白,求指教!
PHP中文网
PHP中文网 2017-04-10 16:13:16
[PHP讨论组]

首先,我定义个类,实现单例模式:(这里是简单一写,就是个最基本的单例)

class Demo
{
    public static $instance;
    
    private function __Construct()
    {
        //TODO
    }
    
    public static function getInstance()
    {
        if(!self::$instance){
            self::$instance = new static();
        }
        
        return self::$instance;
    }
    
    public function call()
    {
        //其他方法
    }
}

下面有两种方式实例化类:

1.在需要用的地方

$aa = Demo::getInstance();
$bb = Demo::getInstance();
$cc = Demo::getInstance();

这样调用肯定是没问题的,一般情况也是这样初始化。

2.定义一个普通类,写个函数初始化,保存在一个静态变量,如下:

类:

class Demo{
    public function __Construct(){
        //TODO
    }
}

函数:

function get_obj(){
    static $obj;
    if($obj){
        return $obj;
    }else{
        $obj = new Demo;
        return $obj;
    }
}

在需要调用的地方这样写:

$obj = get_obj();
$obj->call();

$obj2 = get_obj();
$obj2->call();

$obj3 = get_obj();
$obj3 = get_obj();

这样不是也只实例化一次这个类吗?

PHP中文网
PHP中文网

认证0级讲师

全部回复(13)
阿神

静态变量可以定义在函数里或者类属性,两种写法都可以,但前者封装性会比较好些。

PHP中文网

楼主给了我启发,过程式编程中,在全局函数内用静态变量存储数据库连接实现单例:

<?php
// config.php
$app = array(
    'db_host'        => '127.0.0.1',
    'db_username'    => 'root',
    'db_password'    => '',
    'db_name'        => 'mysql'
);
// functions.php
function cn_db() {
    global $app;
    static $mysqli;
    if ($mysqli) {
        return $mysqli;
    } else {
        $mysqli = new mysqli($app['db_host'], $app['db_username'], $app['db_password'], $app['db_name']);
        return $mysqli;
    }
}
function cn_foo1() {
    $mysqli = cn_db();
    return $mysqli->query('select user,host from user where user = \'root\'')->fetch_all();
}
function cn_foo2() {
    $mysqli = cn_db();
    return $mysqli->query('select user,host from user')->fetch_all();
}
// controller + view
print_r(cn_foo1());
print_r(cn_foo2());
ringa_lee

$instance = new static(); (楼主这行代码写错了),感觉单例模式就是个思想,你第一个例子就是面向对象,第二个例子既有面向对象,又有面向过程。

单例模式是指整个应用中类只有一个对象实例的设计模式。

迷茫

第二种方式是单例吗?

class Demo {
    public function __construct(){
        //TODO
    }
}
function get_obj() {
    static $obj;
    if($obj){
        return $obj;
    }else{
        $obj = new Demo;
        return $obj;
    }
}

var_dump(get_obj());
var_dump(new Demo);

得出的是2个不同的实例

巴扎黑

你那样写不符合面向对象思想的,而且你的单例模式也不完整,没有考虑到clone和extends的情况。可以参考下http://www.xtwind.com/design-pattern-singleton.html

阿神

单例模式,是一种设计思想实现方式

巴扎黑

单例首先是面向对象里面的概念,
第二种就不是面向对象

伊谢尔伦

get_obj()那种写法无法保证外部不new Demo对象

天蓬老师

单例的要义第一:私有化构造方法,当然PHP还提供了clone魔术方法,也要私有化。这样做事为了保证外部实例化对象的入口只有一个,只能是我们暴露出来的静态函数getInstance()

巴扎黑

完善的单例应该是这样的

class Foobar {
    static private $instance;
    
    // 禁止外部new Foobar
    private function __construct() {
    }
    
    // 禁止clone $foobar
    private function __clone() {
    }
    
    static public function getInstance() {
        retrun self::$instance
            ?: (self::$instance = new self);
    }
}

如果还要考虑到继承的话

class Foo {
    static private $instances = [];

    protected function __construct() {
    }

    final private function __clone() {
    }

    final static public function getInstance() {
        $class = get_called_class();

        if (!isset(self::$instances[$class])) {
            self::$instances[$class] = new static;
        }

        return self::$instances[$class];
    }
}

class Bar extends Foo {

}

$foo = Foo::getInstance();
$bar = Bar::getInstance();

get_obj()那种写法,也可以达到目的,但无法禁止new和clone,也就无法做到真正的单例

热门教程
更多>
最新下载
更多>
网站特效
网站源码
网站素材
前端模板
关于我们 免责申明 意见反馈 讲师合作 广告合作 最新更新 English
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送
PHP中文网APP
随时随地碎片化学习

Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号