加载类、设计模式(单例模式和工厂模式)

 

加载类

1.加载类的方法

  加载类可以使用include、require、require_once三种中的任意一种,每个关键字都有两种方法,但是这种方法的缺点是需要加载多少个php文件,就要写多少个加载类的方法。一般也就需要加载一两个类,所以这种写法很常用。

//include("./Ren.class.php");
//include "./Ren.class.php";

//require("./Ren.class.php");
//require "./Ren.class.php";

//require_once("./Ren.class.php");
//require_once "./Ren.class.php";

 

 自动加载类:

  如果要使用自动加载类,要满足下列2个条件

  1.所有文件都放在同一个目录下;

  2.所有类文件的命名规则一致;

//使用条件:1.所有文件都放在同一个目录下;2.所有类文件的命名规则一致。
/*function __autoload($classname)//自动加载类的方法,classname代表类名。
{
    require $classname.".class.php";//可以用第一种加载类的任意一种方法。
}

例如:网文件里加载类名为Ren的类 function __autoload($Ren)
{
  require $Ren.".class.php";//代表加载了class.php文件下的Ren类。类名后面加上文件名个文件格式。
}

$r = new Ren();
*/

 

设计模式

1.单例模式

  类的计划生育:类智能造出一个对象来,不能多造,更不能随便取造。

  单例模式又称为职责模式,它用来在程序中创建一个单一功能的访问点,通俗地说就是实例化出来的对象是唯一的。
  所有的单例模式至少拥有以下三种公共元素:
  1. 它们必须拥有一个构造函数,并且必须被标记为private
  2. 它们拥有一个保存类的实例的静态成员变量
  3. 它们拥有一个访问这个实例的公共的静态方法
  单例类不能再其它类中直接实例化,只能被其自身实例化。它不会创建实例副本,而是会向单例类内部存储的实例返回一个引用。

例题:

class Single {
    private $name;//声明一个私有的实例变量
    private function __construct(){//声明私有构造方法为了防止外部代码使用new来创建对象。
    
    }

    static public $instance;//声明一个静态变量(保存在类中唯一的一个实例)
    static public function getinstance(){//声明一个getinstance()静态方法,用于检测是否有实例对象
        if(!self::$instance) self::$instance = new self();
        return self::$instance;
    }

    public function setname($n){ $this->name = $n; }
    public function getname(){ return $this->name; }
}


$oa = Single::getinstance();
$ob = Single::getinstance();
$oa->setname('hello world');
$ob->setname('good morning');
echo $oa->getname();//good morning
echo $ob->getname();//good morning

 

//从类外部控制
/*$dx;//定一个变量$dx
class Dog//Dog类
{
    
}
if(empty($dx))//判断变量是否为空,为空返回true,不为空返回flase。
{
    $dx = new Dog();//如果变量是空的就造一个对象给$dx。
}
else
{
    $a = $dx;//如果不为空,就把$dx给$a。
}*/
//这样这个界面就不能多造了,但是别的类还可以继续多造,比如$b=new Dog(),$c=new Dog().从类外部控制不能控制全部的,所以要转到内部控制。

 

//从类内部控制

内部控制的思路
//1.让该类在外界无法造对象,使用private.
//2.让外界可以造一个对象,做一个方法返回对象。
//3.在类里面通过静态变量控制返回对象只能有一个。

class Dog
{
  state $dx;

  public $test;

  private function __construct()//为了不让外界访问,先把访问模式变为私有的private,外界无法new对象了。造下面的DuiXiang方法内部访问,返回对象。

  {

  }

  static function Duixiang()//做一个方法,调一下就返回一个对象。外部无法调用此方法,先把此方法变为静态的,不用造对象访问,直接用类名来调用。但是类名每调用一次都会返回一个对象,还是起不到目标效果。做一个成员变量$dx,用来存储对象。
  {
    //return new Dog();//返回一个对象
    if(empty(self::$dx))//先判断$dx这个变量是否为空
    {
      self::$dx = new Dog();//如果变量$dx是空的,就造一个对象给$dx
    }
    return self::$dx;//如果变量$dx不为空,就不用造了,什么都不执行,直接返回$dx。
  }

}

$a = Dog::DuiXiang();//造一个对象$a

$b= Dog::DuiXiang();//造一个对象$b

$b->test="hello";


echo $a->test;//输出结果为hello和$b的一样。

虽然造了两个对象,但是是同一个对象,无论调多少,也都是一个。

 

工厂模式

//写一个父类用子类去继承父类比较安全

abstract class Yunsuan//做一个抽象类
{
        public $a;//变量a 
        public $b;//变量b
        function Suan()//方法Suan
        {
            
        }
}

 class Jia extends Yunsuan//做一个YunSuan的子类
{
    function Suan()//对Suan的方法重写
    {
        return $this->a+$this->b;
    }
}

class Jian extends Yunsuan//修改减法的功能只需要修改这个文件就可以
{
    function Suan()
    {
        return $this->a-$this->b;
    }
}

class Cheng extends Yunsuan//需要扩展功能的话直接再新写一个文件就可以,去掉一个功能直接删除相应文件就可以。
{
    function Suan()
    {
        return $this->a*$this->b;
    }
}
//$j = new Jia9);//箱实现加法就直接写一个加法对象,调用里面的Suan方法就可以了。
//$j->Suan();



//工厂类,生产对象。给不同的参数,生产不同的对象。
class GongChang
{
    static function ShengChan($f)//为了比较方便调用,做成静态的,直接用类名调用。
    {
        switch($f)
        {
            case "+"://如果是+,就返回Jia的对象。
                return new Jia;
                break;
            case "-":
                return new Jian;//如果是-,就返回Jian的对象
                break;
            case "*":
                return new Cheng;
                break;
        }
    }
}

$r = GongChang::ShengChan("+");//实行加法运算,就传一个“+”到类GongChang里面,然后就会返回一个new Jia,也就是加法的对象对象(YunSuan的子类)
$r->a=10;
$r->b=5;
echo $r->Suan();

 

上面的内容就是工厂模式,工厂模式就是指有一个工厂类,根据给的参数的不同生产出不同的对象。一般来说生产的对象有一个相同父类的,就比如说上面的做的那些类,有一个YunSuan的父类,Jia、Jian、Cheng三个子类。然后工厂类就返回这三个子类的任意一个,根据参数的不同返回相应的参数。

 

 

面向对象的六大原则

OOP基本上有6大原则,而实际上都是互补的,也就是说一些原则需要利用另一些原则来实现自己。6大原则如下:

1) Open-Close Principle(OCP),开-闭原则,讲的是设计要对扩展有好的支持,而对修改要严格限制。这是最重要也是最为抽象的原则,基本上我们所说的Reusable Software既是基于此原则而开发的。其他的原则也是对它的实现提供了路径。

2) Liskov Substituition Principle(LSP),里氏代换原则,很严格的原则,规则是“子类必须能够替换基类,否则不应当设计为其子类。”也就是说,子类只能去扩展基类,而不是隐藏或覆盖基类。


3) Dependence Inversion Principle(DIP),依赖倒换原则,“设计要依赖于抽象而不是具体化”。换句话说就是设计的时候我们要用抽象来思考,而不是一上来就开始划分我需要哪些哪些类,因为这些是具体。这样做有什么好处呢?人的思维本身实际上就是很抽象的,我们分析问题的时候不是一下子就考虑到细节,而是很抽象的将整个问题都构思出来,所以面向抽象设计是符合人的思维的。另外这个原则会很好的支持OCP,面向抽象的设计使我们能够不必太多依赖于实现,这样扩展就成为了可能,这个原则也是另一篇文章《Design by Contract》的基石。

4) Interface Segregation Principle(ISP),接口隔离原则,“将大的接口打散成多个小接口”,这样做的好处很明显,我不知道有没有必要再继续描述了,为了节省篇幅,实际上我对这些原则只是做了一个小总结,如果有需要更深入了解的话推荐看《Java与模式》,MS MVP的一:本巨作!^_^

5) 单一职责:一个类的功能尽量单一,降低耦合

6) Law of Demeter or Least Knowlegde Principle(LoD or LKP),迪米特法则或最少知识原则,这个原则首次在Demeter系统中得到正式运用,所以定义为迪米特法则。它讲的是“一个对象应当尽可能少的去了解其他对象”。也就是又一个关于如何松耦合(Loosely-Coupled)的法则。

好了,以上是6大原则(或法则)的介绍,对这些原则的深入研究正是如何得到设计模式的道路。在进行了深入了解后我们就可以开始看看设计模式了,设计模式正是对这些法则的应用,著名的设计模式有四人帮(Gang of Four,GoF)的23个模式,除此之外还有很多其他的一些著名模式,大家可以慢慢研究,如果能自己产出一两个模式的话那就太好了,证明你也是高手了!^_^

 

posted @ 2016-10-27 19:31  Strive-count  阅读(383)  评论(0编辑  收藏  举报