Maison >développement back-end >tutoriel php >看看这个类的实例化方式能否实现“单例模式”?

看看这个类的实例化方式能否实现“单例模式”?

WBOY
WBOYoriginal
2016-06-06 20:26:121329parcourir

比如我有一个类,类名是A(这个类本身不是实现单例模式的,看后面实例化这个类的方式):

<code>class A(){
    public $str = '这是属性';
    public $str2 = '属性';
    private $str3 = '私有属性';
    
    public function __construct($a,$b,$c){
        //构造方法
    }
    
    public function func1(){
        //类的方法
    }
    
    public function func2(){
        //类的方法
    }
}</code>

还有一个全局函数:

<code>function getObj(){
    require PATH.'a.class.php';//引入上面定义的class A的文件
    static $instance;//定义静态变量
    if($instance){
        return $instance;
    }
    $instance = new A('1','2','3');
    return $instance;
}</code>

然后我要在其他地方调用类A的实例,并且可能需要一次请求调用多次,通过getObj方法来得到这个类的实例,因为是返回的静态变量,其实在第一次创建这个类的实例后,后面调用都是直接用的这个静态变量的实例,并没有重新new一个新的实例,这种方式算不算单例模式??如果不算,主要是哪里不同会有什么隐患??十分感谢!

回复内容:

比如我有一个类,类名是A(这个类本身不是实现单例模式的,看后面实例化这个类的方式):

<code>class A(){
    public $str = '这是属性';
    public $str2 = '属性';
    private $str3 = '私有属性';
    
    public function __construct($a,$b,$c){
        //构造方法
    }
    
    public function func1(){
        //类的方法
    }
    
    public function func2(){
        //类的方法
    }
}</code>

还有一个全局函数:

<code>function getObj(){
    require PATH.'a.class.php';//引入上面定义的class A的文件
    static $instance;//定义静态变量
    if($instance){
        return $instance;
    }
    $instance = new A('1','2','3');
    return $instance;
}</code>

然后我要在其他地方调用类A的实例,并且可能需要一次请求调用多次,通过getObj方法来得到这个类的实例,因为是返回的静态变量,其实在第一次创建这个类的实例后,后面调用都是直接用的这个静态变量的实例,并没有重新new一个新的实例,这种方式算不算单例模式??如果不算,主要是哪里不同会有什么隐患??十分感谢!

接下去就要是PHP7的时代了,你的代码还停留在看上去像是5.x早期时代的PHP代码,本身代码应该可以跑通,你的使用方式从宽泛的定义上来讲,勉强可以算单例,但是整体感觉非常奇怪。并且如别的回答提出的,如果别人不知道getObj函数的话,很有可能就自己new了,你的获取对象的函数无法对别人起到约束作用

既然你的构造函数依赖三个参数,这是我对于这份代码的建议实现

<code>class A(){
    public $str = '这是属性';
    public $str2 = '属性';
    private $str3 = '私有属性';
    
    private static $pool = [];
    
    public static function getInstance($a, $b, $c)
    {
        /**
         * 这里假设一种情况是需要这三个参数这个对象才唯一
         * 并且这三个参数都是可以转换为字符串类型的
         *(也就是说不能是stream,array,或者没有__toString()实现的object)
         */
        $key = implode('_', func_get_args()); 

        if (!isset(self::$pool[$key])) {
            /**
             * 关于这里使用static关键字详情请看后期静态绑定
             * http://php.net/manual/zh/language.oop5.late-static-bindings.php
             * 如果你的PHP还在用5.3以前的版本,请使用
             * self::$pool[$key] = new self($a, $b, $c);
             */
            self::$pool[$key] = new static($a, $b, $c); 
        }
        
        return self::$pool[$key];
    }
    
    protected function __construct($a, $b, $c)
    {
        //构造方法,没有使用private是考虑如果有子类继承的情况
    }
    
    public function func1()
    {
        //类的方法
    }
    
    public function func2()
    {
        //类的方法
    }
}</code>

PS 另外建议题主去了解一下PSR规范和autoload

会报错的,而且这种写法很糟糕

其实不算。原因如下:
1.假设是多线程的环境,多个线程同时进入会不会同时产生多个对象呢,所以需要你进行冗余处理。我不知道php是怎么做的,但是Java可以通过sync关键字来搞定同步。
2.你构造函数是公有的,假设别人在你的代码上二次开发,但他不知道你的这个getobj。他就会自己new一个。这样也会有问题。

Déclaration:
Le contenu de cet article est volontairement contribué par les internautes et les droits d'auteur appartiennent à l'auteur original. Ce site n'assume aucune responsabilité légale correspondante. Si vous trouvez un contenu suspecté de plagiat ou de contrefaçon, veuillez contacter admin@php.cn