php示例详解Constructor Prototype Pattern 原型模式


Posted in PHP onOctober 15, 2015

原型模式中主要角色

抽象原型(Prototype)角色:声明一个克隆自己的接口
具体原型(Concrete Prototype)角色:实现一个克隆自己的操作

当一个类大部分都是相同的只有部分是不同的时候,如果需要大量这个类的对象,每次都重复实例化那些相同的部分是开销很大的,而如果clone之前建立对象的那些相同的部分,就可以节约开销。

针对php的一种实现方式就是__construct()和initialize函数分开分别处理这个类的初始化,construct里面放prototype也就是公共的部分,initialize里面是每个对象特殊的部分。这样我们先建立一个类不initialize,以后每次clone这个类再进行initialize就可以了。

 在zend framework官方手册里面提到了这个http://framework.zend.com/manual/2.0/en/user-guide/database-and-models.html,但是没有细讲,下面我来分析一下

一、引入

在zf2的model里面有一个albumTable类,相当于一个操作数据库动作的助手类,里面用到了tablegateway。

为了每次初始化albumtable都是相同的一个类,将初始化工作放到了根目录的module.php文件的getServiceConfig(),其中用到工厂模式,并且通过回调函数,当每次ServiceManager($sm)需要实例化一个对象的时候会自动调用创建一个alumTable。下面代码我们可以看出,创建一个albumTable还需要用相同的方式创建一个AlbumTableGateWay,这个类就用到了我们所要讲的原型模式。

二、代码详解

public function getServiceConfig()
  {
    return array(
      'factories' => array(
        'Album\Model\AlbumTable' => function($sm) {
          $tableGateway = $sm->get('AlbumTableGateway');
          $table = new AlbumTable($tableGateway);
          return $table;
        },
        'AlbumTableGateway' => function ($sm) {
          $dbAdapter = $sm->get('Zend\Db\Adapter\Adapter');
          $resultSetPrototype = new ResultSet();
          $resultSetPrototype->setArrayObjectPrototype(new Album());//这个就是一个不变的原型
          return new TableGateway('album', $dbAdapter, null, $resultSetPrototype);//传入到TableGateWay的构造函数中去
        },
      ),
    );
  }

注意并不是TableGateWay运用了原型模式而是ResultSet这个类运用了。每当tablegateway调用select()或者insert()等方法的时候都会建立一个ResultSet用来表示结果,这些ResultSet中公共部分被clone,而独特的部分类如data就会被initialize。

三、更多代码示例

为了更清晰得了解这个原型,我们先抛开zend这个大框架,看一个完整的代码示例。示例来自

<a href="http://ralphschindler.com/2012/03/09/php-constructor-best-practices-and-the-prototype-pattern">PHP Constructor Best Practices And The Prototype Pattern</a>

这篇文章关于prototype pattern的部分前半部分其实是混杂怎样在构造函数中运用继承来提高扩展性,两个模式看起来可能不太好理解,我们直接看最后的代码关于prototype pattern的部分。

<?php
//框架中很常见的adapter类,用来适配各种数据库,封装一些基本数据库连接操作。
//相当于上面代码中的adapter类
class DbAdapter {
  public function fetchAllFromTable($table) {
    return $arrayOfData;
  }
}
//运用prototype pattern的类,注意construct和initialize是分开的
//相当于上面zend 代码里面的ResultSet类
class RowGateway {
  public function __construct(DbAdapter $dbAdapter, $tableName) {
    $this->dbAdapter = $dbAdapter;
    $this->tableName = $tableName;
  }
  public function initialize($data) {
    $this->data = $data;
  }
  /**
   * Both methods require access to the database adapter
   * to fulfill their duties
   */
  public function save() {}
  public function delete() {}
  public function refresh() {}
}
//相当于上面代码中的TableGateway类,关于gateway可以具体去了解一下。
class UserRepository {
  public function __construct(DbAdapter $dbAdapter, RowGateway $rowGatewayPrototype = null) {
    $this->dbAdapter = $dbAdapter;
    $this->rowGatewayPrototype = ($rowGatewayPrototype) ? new RowGateway($this->dbAdapter, 'user')
  }
  public function getUsers() {
    $rows = array();
    foreach ($this->dbAdapter->fetchAllFromTable('user') as $rowData) {
      $rows[] = $row = clone $this->rowGatewayPrototype;
      $row->initialize($rowData);
    }
    return $rows;
  }
}

这几个类其实和上面zend代码中的类是对应的

Dbadapter -- adpater

RowGateWay -- ResultSet

UserRepository - TableGateWay

具体看代码中的注释。

这里的RowGateWay可以很明显的看出在getusers中需要大量的实例化,那么原型模式就是很必要的了。

下面是运用这个类的代码

class ReadWriteRowGateway extends RowGateway {
  public function __construct(DbAdapter $readDbAdapter, DbAdapter $writeDbAdapter, $tableName) {
    $this->readDbAdapter = $readDbAdapter;
    parent::__construct($writeDbAdapter, $tableName);
  }
  public function refresh() {
    // utilize $this->readDbAdapter instead of $this->dbAdapter in RowGateway base implementation
  }
}
// usage:
$userRepository = new UserRepository(
  $dbAdapter,
  new ReadWriteRowGateway($readDbAdapter, $writeDbAdapter, 'user')
);
$users = $userRepository->getUsers();
$user = $users[0]; // instance of ReadWriteRowGateway with a specific row of data from the db

以上内容是小编给大家介绍的php示例详解Constructor Prototype Pattern 原型模式,希望大家喜欢。

PHP 相关文章推荐
AJAX for PHP简单表数据查询实例
Jan 02 PHP
PHP中提问频率最高的11个面试题和答案
Sep 02 PHP
php程序总是提示验证码输入有误解决方案
Jan 07 PHP
smarty模板引擎从php中获取数据的方法
Jan 22 PHP
PHP之预定义接口详解
Jul 29 PHP
PHP设计模式之简单投诉页面实例
Feb 24 PHP
Yii2超好用的日期和时间组件(值得收藏)
May 05 PHP
php+ajax无刷新上传图片的实现方法
Dec 06 PHP
php通过header发送自定义数据方法
Jan 18 PHP
浅谈Laravel核心解读之Console内核
Dec 02 PHP
php中的依赖注入实例详解
Aug 14 PHP
Laravel-添加后台模板AdminLte的实现方法
Oct 08 PHP
PHP经典面试题之设计模式(经常遇到)
Oct 15 #PHP
PHP面试题之文件目录操作
Oct 15 #PHP
php cli配置文件问题分析
Oct 15 #PHP
PHP+Mysql+jQuery实现发布微博程序 php篇
Oct 15 #PHP
10个php函数实用却不常见
Oct 13 #PHP
PHP实现连接设备、通讯和发送命令的方法
Oct 13 #PHP
PHP如何通过传引用的思想实现无限分类(代码简单)
Oct 13 #PHP
You might like
dedecms中使用php语句指南
2014/11/13 PHP
PHP实现递归目录的5种方法
2016/10/27 PHP
PHP addAttribute()函数讲解
2019/02/03 PHP
PHP 实现文件压缩解压操作的方法
2019/06/14 PHP
Laravel框架实现文件上传的方法分析
2019/09/29 PHP
如何在Laravel之外使用illuminate组件详解
2020/09/20 PHP
Javascript实现的鼠标经过时播放声音
2010/05/18 Javascript
Javascript中正则表达式的全局匹配模式分析
2011/04/26 Javascript
multiSteps 基于Jquery的多步骤滑动切换插件
2011/07/22 Javascript
JavaScript/jQuery 表单美化插件小结
2012/02/14 Javascript
jQuery动态添加 input type=file的实现代码
2012/06/14 Javascript
Node.js 学习笔记之简介、安装及配置
2015/03/03 Javascript
javascript中setTimeout使用指南
2015/07/26 Javascript
Spring MVC中Ajax实现二级联动的简单实例
2016/07/06 Javascript
15个非常实用的JavaScript代码片段
2016/12/18 Javascript
ES6新特性二:Iterator(遍历器)和for-of循环详解
2017/04/20 Javascript
Three.js利用Detector.js插件如何实现兼容性检测详解
2017/09/26 Javascript
JS中的回调函数实例浅析
2018/03/21 Javascript
angularjs获取到My97DatePicker选中的值方法
2018/10/02 Javascript
VuePress 中如何增加用户登录功能
2019/11/29 Javascript
优化Vue中date format的性能详解
2020/01/13 Javascript
[02:04]2014DOTA2国际邀请赛 BBC小组赛第三天总结
2014/07/12 DOTA
Python抽象和自定义类定义与用法示例
2018/08/23 Python
python使用PIL剪切和拼接图片
2020/03/23 Python
实例讲解Python 迭代器与生成器
2020/07/08 Python
实例讲解CSS3中的border-radius属性
2015/08/18 HTML / CSS
德国旅游网站:weg.de
2018/06/03 全球购物
.net面试题
2016/09/17 面试题
一名老师的自我评价
2014/02/07 职场文书
计算机系本科生求职信
2014/05/31 职场文书
住房租房协议书
2014/08/20 职场文书
资料员岗位职责
2015/02/10 职场文书
中秋节慰问信
2015/02/15 职场文书
2015最新婚礼主持词
2015/06/30 职场文书
2016入党积极分子心得体会
2016/01/06 职场文书
python创建字典及相关管理操作
2022/04/13 Python