zoukankan      html  css  js  c++  java
  • php设计模式总结-单件模式

    一、单件模式

    英文叫做sington。其他语言中有叫做单例模式,其实都是一样的道理。保证只会出现单个实例,所以是单例。翻译成单件,永远只会产生一件,呵呵。

    还有翻译成单元素模式。其实关键是看这个英文比较好。英文是sington,统一是使用这个单词。



    单件模式的目的我理解如下:

    避免重复创建(实例化)对象,已经有现成的实例就用现成的。

    减少资源的浪费(因为创建多个实例,浪费内存,完全没必要),单件模式保证了每时每刻引用的都是同一个实例。


    为什么同时创建多个实例会引起逻辑上的错误呢?

    $obj1
    $obj2
    多个实例。可能会覆盖掉里面的静态static变量吗? 不是这样子的。

    其实是因为我目前还没遇到更加严重的问题。目前是简单的应用。


    二、我觉得单件模式实践的注意点在下面几个方面

    1、不要使用全局变量来保存实例值。因为全局变量在任何地方都可以被访问和修改,这就意味着可能会被其他代码给破坏掉值,这样就达不到永远是同一

    个实例的效果。


    2、使用static静态变量。这样只能函数内部访问。解决了全局变量被破坏的风险。

    我觉得这是很多要做到实例唯一的一个关键部分。像框架中为保证所有对类实例的引用是唯一一个,都是将实例保存在static变量中。这样子下回调用的

    时候也是同一个实例。不会重复创建。

    抓住了这个精髓,我觉得是可以变化的。并不一定要遵循设计模式书中的做法。因为目标是相同的。技巧可以不同。



    3、一般将类的__construct()构造函数标识为private,这样就是避免程序员直接实例化这个类。根据每种语言的特点,加上private关键词,程序员new一

    个对象,就会报错。这种技巧是一种辅助手段。为保证只有一个类实例做辅助方案的。核心还是在于第二点的static关键字。

    只要程序员约定好,这个辅助手段其实可以没有仍然能够做到单件。不是为设计模式而设计。了解实现目标才是关键的。


    我在想,可以使用protected来替代吗?

    目标就是,要禁止使用new来实例化这个函数。当实例化一个类的时候,默认会去执行构造函数,而加上protected和private关键字的成员,
    都同样不能在类外部调用的。所以使用protected也是可行的。

    但为什么要使用private呢?还有个好处,可以避免被继承的子类所重写,覆盖掉方法的内容。因为加上protected标识的成员是能够被子类给重写的。
    既然对构造函数加上了private,那就意味着子类是不能继承这个类的。了解这个特性设计的时候就要考虑,无子类继承它的概念。








    4、代码实践

    class test
    {


    static $_instance = false;

    private function __construct()
    {

          /*一般将构造函数加上private关键字,这样子避免直接使用外部直接new来实例对象,当然内部使用new来创建是不会影响的*/
    }


    function get_instance()
    {

              if(self::$_instance==false && !is_object(self::$_instance)){
           
             self::$_instance = new test();
          }
          return self::$_instance;

    }



    }



    实际项目开发中,有个变体是,创建a、b、c的实例都需要通过一个公共的方法来调用,这样子可以实现单件模式。

    类似于thinkphp等框架中的。

    像下面是phpcms中的


    pc_base::load_app_calss('test');


    load_app_class($class_name)
    {

    static $class_array = array();

    if(isset($class_array[$class_name]) && is_object($class_array[$class_name])) )
    return $class_array[$class_name];
    }else{
    //这里可能还要有代码载入这个类文件,根据实际而定。可以是去默认一个文件夹夹中载入。也可以认为调用这个方法的前提是类文件要载入进来
    $class_array[$class_name] = new $class_array[$class_name];
    return $class_array[$class_name];

    }


    其实可以避免创建很多数据库链接。写到这里,我想起了mysql对于同一组参数进行的mysql_connect()连接,是不会重新建立连接的。php手册中对这个函

    数的解释如下:

    如果用同样的参数第二次调用 mysql_connect(),将不会建立新连接,而将返回已经打开的连接标识。

     

    其实呢,只是mysql_connect这个函数做了可复用了。不讨论数据库连接方面。实例化其他的类,也需要创建大量的实例。占有资源。是指同一次执行的代

    码过程中才能起到节省资源的效果

     

    比如a.php的代码过程如下:

    $class = test::get_instance();//得到这个test这个类的实例

    $class->get_name();

    get_count_number();//假设这个函数里面又需要用到那个类,则又需要进行实例化,如果统一调用get_instance()来获取实例,则前面得到的实例是可以

    复用的。

     






    三、单态模式(monostate)



    1、单件模式还有一种变体:就是类的单件模式,也就是monostate模式。MonoState的意思就是"单一的状态"。也就是常说的单态。实现的目标为:所有实

    例都是共享类中同一个值。

    monostate的设计目标为:实现多个实例可以共享变量(类里面的属性),成为单态,尽管存在多个实例,但实例中的变量的最终只会有一个状态(可以理解为

    一个值),不会出现多个值(也就是每个实例里面的变量都是不同的值)。

    2、它与单件的区别为:

    单件是将构造函数声明为private,来保证只有一个实例。而单态则不需要。它关注的侧重点是最终只有一个数值,而用户实例化多少类,不是它所关心的


    MonoState并不限制创建对象的个数,但是它的状态却只有一个状态。



    3、monostate模式实践


    实践要点:把类里面的变量(属性)标识为static即可


    <?php

    class test
    {

    static $_state = array();


    function set($key,$value)
    {
      self::$_state[$key]= $value;
    }


    function get($key)
    {

       return self::$_state[$key];

    }



    }


    $obj = new test();

    $obj->set('name','wangtao');

    $obj->set('sex','male');


    echo $obj->get('name');//得到结果是wangtao


    //再次实例化一次,看访问对象的成员,是否得到一样的数据。
    $obj2 = new test();

    echo $obj2->get('name');//输出wangtao

     

    //再次新创建一个实例$obj2,访问name这个变量,数据是共享的,所以输出还是wangtao。当然使用set()把值改变了,其他实例也会访问到改变后的值。



    总结:实现monostate模式,具体实现有多种办法,只要达到共享数据的目的就ok。比如使用$_GLOABS[]全局变量,把数据保存在全局变量中,然后放到类

    成员中也可以,《php设计模式》这本书就是使用这种形式实现。使用静态变量(static关键字)也可以。上面使用的就是静态变量的方式。我觉得使用

    static方式更加直观易懂









    四、思考:sington与monostate能混合一起实现吗?

    既然sington模式可以避免创建多个实例。而monostate是关注多个实例之间共享数据。

    那么有没有种办法让两者混合呢。

    也就是说:我构造一个类,既能够达到单件的效果,也能实现monostate的效果。开玩笑玩玩,呵呵,加深深入理解。

    我觉得,单件关注的是实例化一个类。monostate关注的状态的一致性。其实两者是不相容的。

    如果实现了单件模式。那么就不存在多个实例对象存在。既然都是调用同一个实例,这样子里面的成员变量肯定是共享的,因为使用的是同一个实例的成员

    。为此我特意做试验,如下:


    class test
    {

    static $_state;//实现单态,就是将里面变量定义为static即可,现在这个类实现了monostate模式
    static $_instance = false;

    private function __construct()
    {


    }

    /*
    实现单例模式
    */
    function get_instance()
    {

           if(self::$_instance==false && !is_object(self::$_instance)){
           
             self::$_instance = new test();
          }
          return self::$_instance;

    }



    }

    $obj1 = test::get_instance();
    $obj1->_state = 20;

    $obj2 = test::get_instance();//因为这里引用的还是同一个实例,所以下面输出属性的值,还是前面更改的20

    echo $obj2->_state;






     

     



    以上是给自己总结用。不正确之处欢迎指正。


  • 相关阅读:
    boost编译中的细节问题
    geos编译问题
    安装pytorch的细节记录
    Qt学习-模仿Qt实现一个colorbutton
    BOOST内存管理-intrusive_ptr
    GEOS使用记录
    matlab添加永久路径
    关于浮点数的取值范围以及精度的问题
    vs2010中使用命令行参数
    sprintf fprintf EOF scanf 的返回值 深拷贝与浅拷贝
  • 原文地址:https://www.cnblogs.com/wangtao_20/p/3593076.html
Copyright © 2011-2022 走看看