PHP中常用的三种设计模式详解【单例模式、工厂模式、观察者模式】

 更新时间:2019年8月17日 19:36  点击:276

本文实例讲述了PHP中常用的三种设计模式。分享给大家供大家参考,具体如下:

PHP中常用的三种设计模式:单例模式、工厂模式、观察者模式

1.单例模式

为何要使用PHP单例模式?

多数人都是从单例模式的字面上的意思来理解它的用途, 认为这是对系统资源的节省, 可以避免重复实例化, 是一种”计划生育”. 而PHP每次执行完页面都是会从内存中清理掉所有的资源. 因而PHP中的单例实际每次运行都是需要重新实例化的, 这样就失去了单例重复实例化的意义了. 单单从这个方面来说, PHP的单例的确有点让各位失望. 但是单例仅仅只有这个功能和应用吗? 答案是否定的。

  1. php的应用主要在于数据库应用, 所以一个应用中会存在大量的数据库操作, 在使用面向对象的方式开发时(废话), 如果使用单例模式, 则可以避免大量的new 操作消耗的资源。

  2. 如果系统中需要有一个类来全局控制某些配置信息, 那么使用单例模式可以很方便的实现. 这个可以参看zend Framework的FrontController部分。

  3. 在一次页面请求中, 便于进行调试, 因为所有的代码(例如数据库操作类db)都集中在一个类中, 我们可以在类中设置钩子, 输出日志,从而避免到处var_dump, echo。

一个单例类应包括以下几点:

和普通类不同,单例类不能被直接实例化,只能是由自身实例化。因此,要获得这样的限制效果,构造函数必须标记为private

要让单例类不被直接实例化而能起到作用,就必须为其提供这样的一个实例。因此,就必须要让单例类拥有一个能保存类的实例的私有静态成员变量和对应的一个能访问到实例的公共静态方法。

在PHP中,为防止对单例类对象的克隆来打破单例类的上述实现形式,通常还为其提供一个空的私有__clone()方法。

对于一个类的对象,如果使用“clone运算符”,就会复制出一个和当前对象完全一样的新对象出来,并且,此时还会自动调用该类的魔术方法:__clone()(只要该类中有该方法)。

则要实现单例类,就应该对这个单例类的对象“禁止克隆”,用private来修饰__clone()来实现禁止克隆

单例模式的例子:

<?php
/*** 设计模式之单例模式* $instance必须声明为静态的私有变量* 构造函数和析构函数必须声明为私有,防止外部程序new* 类从而失去单例模式的意义* getInstance()方法必须设置为公有的,必须调用此方法* 以返回实例的一个引用* ::操作符只能访问静态变量和静态函数* new对象都会消耗内存* 使用场景:最常用的地方是数据库连接。* 使用单例模式生成一个对象后,* 该对象可以被其它众多对象所使用。*/
class SingetonBasic {private static $instance; //静态变量要私有化,防止类外修改// other vars..
private function __construct() {  //构造函数私有化,类外不能直接新建对象  // do construct..
}
private function __clone() {}  //在__clone()前用private修饰,用来禁止克隆
public static function getInstance() { //公共的静态方法,public――外部的接口,static――不使用对象而是通过类名访问  
if (!(self::$instance instanceof self)) { //私有静态变量$instance为空    
self::$instance = new self(); //新建为自身的对象,并赋值给私有变量$instance  
}  return self::$instance; //返回私有变量$instance
}// other functions..
}$a = SingetonBasic::getInstance();
$b = SingetonBasic::getInstance();
var_dump($a === $b); 
 //结果为:boolean true//
 ?>
<?php/** * php单例,单例模式为何只能实例化一次*/
class Example{  // 保存类实例在此属性中  
private static $instance;  // 构造方法声明为private,防止直接创建对象  
private function __construct(){    echo 'I am constructed';  }  // singleton 方法  
public static function singleton(){    if (!isset(self::$instance)) {//判断是否以前创建了当前类的实例      
$c = __CLASS__;//获取类名      
self::$instance = new $c;//如果没有创建,实例化当前类,这里实现类只实例化一次    
}    return self::$instance;//返回类的实例  
}  // Example类中的普通方法  
public function bark(){    echo 'Woof!';  }  // 阻止用户复制对象实例  
public function __clone(){    trigger_error('Clone is not allowed.', E_USER_ERROR);  }}// 这个写法会出错,因为构造方法被声明为
private$test = new Example;// 下面将得到Example类的单例对象
$test = Example::singleton();$test->bark();// 下面将得到Example类的单例对象
$test = Example::singleton();$test->bark();// 复制对象将导致一个
E_USER_ERROR.$test_clone = clone $test;
?>

2. 工厂模式

工厂模式在于可以根据输入参数或者应用程序配置的不同来创建一种专门用来实现化并返回其它类的实例的类。

工厂模式的例子:

class FactoryBasic {  public static function create($config) {  }}

比如这里是一个描述形状对象的工厂,它希望根据传入的参数个数不同来创建不同的形状。

<&#63;php// 定义形状的公共功能:获取周长和面积。interface IShape {  function getCircum();  function getArea();}// 定义矩形类class Rectangle implements IShape {  private $width, $height;  public function __construct($width, $height) {    $this->width = $width;    $this->height = $height;  }  public function getCircum() {    return 2 * ($this->width + $this->height);  }  public function getArea() {    return $this->width * $this->height;  }}// 定义圆类class Circle implements IShape {  private $radii;  public function __construct($radii) {    $this->radii = $radii;  }  public function getCircum() {    return 2 * M_PI * $this->radii;  }  public function getArea() {    return M_PI * pow($this->radii, 2);  }}// 根据传入的参数个数不同来创建不同的形状。class FactoryShape {  public static function create() {    switch (func_num_args()) {      case 1:      return new Circle(func_get_arg(0));      break;      case 2:      return new Rectangle(func_get_arg(0), func_get_arg(1));      break;    }  }}// 矩形对象$c = FactoryShape::create(4, 2);var_dump($c->getArea());// 圆对象$o = FactoryShape::create(2);var_dump($o->getArea());

使用工厂模式使得在调用方法时变得更容易,因为它只有一个类和一个方法,若没有使用工厂模式,则要在调用时决定应该调用哪个类和哪个方法;使用工厂模式还使得未来对应用程序做改变时更加容易,比如要增加一种形状的支持,只需要修改工厂类中的create()一个方法,而没有使用工厂模式,则要修改调用形状的代码块。

3. 观察者模式

观察者模式为您提供了避免组件之间紧密耦合的另一种方法。该模式非常简单:一个对象通过添加一个方法(该方法允许另一个对象,即观察者注册自己)使本身变得可观察。当可观察的对象更改时,它会将消息发送到已注册的观察者。这些观察者使用该信息执行的操作与可观察的对象无关。结果是对象可以相互对话,而不必了解原因。

一个简单的示例:当听众在收听电台时(即电台加入一个新听众),它将发送出一条提示消息,通过发送消息的日志观察者可以观察这些消息。

<&#63;php// 观察者接口interface IObserver {  function onListen($sender, $args);  function getName();}// 可被观察接口interface IObservable {  function addObserver($observer);  function removeObserver($observer_name);}// 观察者类abstract class Observer implements IObserver {  protected $name;  public function getName() {    return $this->name;  }}// 可被观察类class Observable implements IObservable {  protected $observers = array();  public function addObserver($observer) {    if (!($observer instanceof IObserver)) {      return;    }    $this->observers[] = $observer;  }  public function removeObserver($observer_name) {    foreach ($this->observers as $index => $observer) {      if ($observer->getName() === $observer_name) {        array_splice($this->observers, $index, 1);        return;      }    }  }}// 模拟一个可以被观察的类:RadioStationclass RadioStation extends Observable {  public function addListener($listener) {    foreach ($this->observers as $observer) {      $observer->onListen($this, $listener);    }  }}// 模拟一个观察者类class RadioStationLogger extends Observer {  protected $name = 'logger';  public function onListen($sender, $args) {    echo $args, ' join the radiostation.<br/>';  }}// 模拟另外一个观察者类class OtherObserver extends Observer {  protected $name = 'other';  public function onListen($sender, $args) {    echo 'other observer..<br/>';  }}$rs = new RadioStation();// 注入观察者$rs->addObserver(new RadioStationLogger());$rs->addObserver(new OtherObserver());// 移除观察者$rs->removeObserver('other');// 可以看到观察到的信息$rs->addListener('cctv');&#63;>


[!--infotagslink--]

相关文章

  • 学习JavaScript设计模式之装饰者模式

    这篇文章主要为大家介绍了JavaScript设计模式中的装饰者模式,对JavaScript设计模式感兴趣的小伙伴们可以参考一下...2016-01-21
  • javascript设计模式之解释器模式详解

    神马是“解释器模式”?先翻开《GOF》看看Definition:给定一个语言,定义它的文法的一种表示,并定义一个解释器,这个解释器使用该表示来解释语言中的句子。在开篇之前还是要科普几个概念: 抽象语法树: 解释器模式并未解释如...2014-06-07
  • JavaScript设计模式之职责链模式

    这篇文章主要介绍了JavaScript设计模式之职责链模式,对设计模式感兴趣的同学,可以参考下...2021-04-25
  • 学习JavaScript设计模式之状态模式

    这篇文章主要为大家介绍了JavaScript设计模式中的状态模式,对JavaScript设计模式感兴趣的小伙伴们可以参考一下...2016-01-12
  • JavaScript设计模式之命令模式

    这篇文章主要介绍了JavaScript设计模式之命令模式,对设计模式感兴趣的同学,可以参考下...2021-04-25
  • 学习JavaScript设计模式之单例模式

    这篇文章主要为大家介绍了JavaScript设计模式中的单例模式,对JavaScript设计模式感兴趣的小伙伴们可以参考一下...2016-01-21
  • 学习JavaScript设计模式之责任链模式

    这篇文章主要为大家介绍了JavaScript设计模式中的责任链模式,对JavaScript设计模式感兴趣的小伙伴们可以参考一下...2016-01-21
  • 学习JavaScript设计模式之代理模式

    这篇文章主要为大家介绍了JavaScript设计模式中的状态模式,对JavaScript设计模式感兴趣的小伙伴们可以参考一下...2016-01-14
  • 学习JavaScript设计模式之观察者模式

    这篇文章主要为大家介绍了JavaScript设计模式中的观察者模式,对JavaScript设计模式感兴趣的小伙伴们可以参考一下...2016-01-21
  • JavaScript中的设计模式 单例模式

    这篇文章主要给大家介绍的是JavaScript中的单例模式,设计模式代表了最佳的实践,通常被有经验的面向对象的软件开发人员所采用。设计模式是软件开发人员在软件开发过程中面临的一般问题的解决方案,需要的朋友可以参考一下...2021-09-25
  • C# 设计模式系列教程-组合模式

    组合模式可以使客户端调用简单,它可以一致使用组合结构或是其中单个对象,简化了客户端代码。...2020-06-25
  • C# 设计模式系列教程-单例模式

    单例模式防止在应用程序中实例化多个对象。这就节约了开销,每个实例都要占用一定的内存,创建对象时需要时间和空间。...2020-06-25
  • 分析设计模式之模板方法Java实现

    所谓模板方法模式,就是一个对模板的应用,就好比老师出试卷,每个人的试卷都是一样的,这个原版试卷就是一个模板,可每个人写在试卷上的答案都是不一样的,这就是模板方法模式。它的主要用途在于将不变的行为从子类搬到超类,去除了子类中的重复代码...2021-06-23
  • Javascript设计模式之原型模式详细

    这篇文章主要介绍了Javascript设计模式之原型模式,原型模式用于在创建对象时,通过共享某个对象原型的属性和方法,从而达到提高性能、降低内存占用、代码复用的效果。下面小编将详细介绍,需要的朋友可以参考下...2021-09-30
  • 详解设计模式中的模板方法模式及在C++中的使用

    这篇文章主要介绍了设计模式中的模板方法模式及在C++中的使用,模板方法将逻辑封装到一个类中,并采取组合(委托)的方式解决这个问题,需要的朋友可以参考下...2020-04-25
  • 深入浅出23种设计模式

    本文给大家详解23种设计模式,理解设计模式有助于在程序开发过程中灵活应用,需要的朋友可以参考下...2020-06-25
  • 实例解析C++设计模式编程中简单工厂模式的采用

    这篇文章主要介绍了C++设计模式编程中简单工厂模式的采用实例,在简单工厂模式中程序往往利用封装继承来降低耦合度,需要的朋友可以参考下...2020-04-25
  • 详解C++设计模式编程中对状态模式的运用

    这篇文章主要介绍了C++设计模式编程中对状态模式的运用,状态模式允许一个对象在其内部状态改变时改变它的行为,对象看起来似乎修改了它的类,需要的朋友可以参考下...2020-04-25
  • PHP设计模式装饰器模式实例

    php面向对象的设计模式中有很多种模式了,今天我们为各位介绍的是装饰器模式的一个学习笔记了,有需要了解php装饰器模式的朋友可以和小编来看看。 我们在使用面向对...2016-11-25
  • 详解设计模式中的Command命令模式及相关C++实现

    这篇文章主要介绍了详解设计模式中的Command命令模式及相关C++实现,命令模式强调调用操作的对象和操作的具体实现者之间的解耦,需要的朋友可以参考下...2020-04-25