加入收藏 | 设为首页 | 会员中心 | 我要投稿 我爱资讯网 (https://www.52junxun.com/)- 科技、建站、经验、云计算、5G、大数据,站长网!
当前位置: 首页 > 站长学院 > PHP教程 > 正文

php接口 PHP设计模式之迭代器模式

发布时间:2022-11-04 11:03:40 所属栏目:PHP教程 来源:
导读:  PHP设计模式之迭代器模式

  一说到这个模式,就不得不提循环语句。在《大话设计模式》中,作者说道这个模式现在的学习意义更大于实际意义,这是为什么呢?当然就是被foreach这货给整得。任何语言都有这种类
  PHP设计模式之迭代器模式
 
  一说到这个模式,就不得不提循环语句。在《大话设计模式》中,作者说道这个模式现在的学习意义更大于实际意义,这是为什么呢?当然就是被foreach这货给整得。任何语言都有这种类似的语法可以方便快捷的对数组、对象进行遍历,从而让迭代器模式从高高在上的23大设计模式中的明星慢慢成为了路人。特别是我们这门PHP语言,PHP的强大之处就在于对于数组的灵活操作,本身就是hashmap的结构,自然会有各种方便的数组操作语法,而foreach也是我们最常用的语句,甚至比for还常用。
 
  Gof类图及解释
 
  GoF定义:提供一种方法顺序访问一个聚合对象中各个元素,而又不需暴露该对象的内部表示

  代码实现
 
  interface Aggregate
  {
      public function CreateIterator();
  }
  class ConcreteAggregate implements Aggregate
  {
      public function CreateIterator()
      {
          $list = [
              "a",
              "b",
              "c",
              "d",
          ];
          return new ConcreteIterator($list);
      }
  }
  首先是聚合类,也就是可以进行迭代的类,这里因为我是面向对象的设计模式,所以迭代器模式针对的是对一个类的内容进行迭代。在这里,其实我们也只是模拟了一个数组交给了迭代器。
 
  interface MyIterator
  {
      public function First();
      public function Next();
      public function IsDone();
      public function CurrentItem();
  }
  class ConcreteIterator implements MyIterator
  {
      private $list;
      private $index;
      public function __construct($list)
      {
          $this->list = $list;
          $this->index = 0;
      }
      public function First()
      {
          $this->index = 0;
      }
      public function Next()
      {
          $this->index++;
      }
      public function IsDone()
      {
          return $this->index >= count($this->list);
      }
      public function CurrentItem()
      {
          return $this->list[$this->index];
      }
  }
  迭代器闪亮登场,主要实现了四个方法来对集合数据进行操作。有点像学习数据结构或数据库时对游标进行的操作。用First()和Next()来移动游标,用CurrentItem()来获得当前游标的数据内容,用IsDone()来确认是否还有下一条数据。所以,这个模式也另称为游标模式。
 
  $agreegate = new ConcreteAggregate();
  $iterator = $agreegate->CreateIterator();
  while (!$iterator->IsDone()) {
      echo $iterator->CurrentItem(), PHP_EOL;
      $iterator->Next();
  }
  客户端直接使用while来进行操作即可。
 
  我们的手机工厂不得了,自己组装了一条生产线,这条生产线主要是做什么的呢?成型机我们已经交给富X康来搞定了,我们这条线就是给手机刷颜色的。当我们把所有已经交货的手机(Aggregate)放到不同的生产线后(Iterator),就会一台一台的帮我们刷上当前生产线的颜色php接口,是不是很强大!!科技不止于换壳,这条线还在,我们就可以再做别的事儿,比如加点挂绳什么的,反正只要能一台一台的通过我就能装上东西,你说好用不好用!!
 
  完整代码:github.com/zhangyue0503/designpatterns-php/blob/master/07.iterator/source/iterator.php
 
  实例
 
  实例还是围绕着我们的短信发送来看。这一次,我们的业务需求是尽快的发一批通知短信给用户,因为活动的时候可不等人啊。在之前我们会使用多个脚本来把用户手机号分成多组来进行发送。现在我们可以用swoole来直接多线程的发送。所要达到的效果其实就是为了快速的把成百上千的短信发完。这个时候我们也会做一些策略,比如数据库里是100条要送的短信,有个字段是发送状态,一个线程正序的发,一个线程倒序的发,当正序和倒序都发送到50条的时候其实已经同步的发完这100条了,不过也有可能会有失败的情况出现,这时,两个线程还会继续去发送那些上次发送不成功的信息,这样能够最大程度的确保发送的效率和到达率。
 
  消息发送迭代器类图
 
  完整源码:github.com/zhangyue0503/designpatterns-php/blob/master/07.iterator/source/iterator-msg.php
 
  list = $list;
          $this->index = 0;
      }
      public function First()
      {
          $this->index = 0;
      }
      public function Next()
      {
          $this->index++;
      }
      public function IsDone()
      {
          return $this->index >= count($this->list);
      }
      public function CurrentItem()
      {
          return $this->list[$this->index];
      }
  }
  // 反向迭代器
  class MsgIteratorDesc implements MsgIterator
  {
      private $list;
      private $index;
      public function __construct($list)
      {
          // 反转数组
          $this->list = array_reverse($list);
          $this->index = 0;
      }
      public function First()
      {
          $this->index = 0;
      }
      public function Next()
      {
          $this->index++;
      }
      public function IsDone()
      {
          return $this->index >= count($this->list);
      }
      public function CurrentItem()
      {
          return $this->list[$this->index];
      }
  }
  interface Message
  {
      public function CreateIterator($list);
  }
  class MessageAsc implements Message
  {
      public function CreateIterator($list)
      {
          return new MsgIteratorAsc($list);
      }
  }
  class MessageDesc implements Message
  {
      public function CreateIterator($list)
      {
          return new MsgIteratorDesc($list);
      }
  }
  // 要发的短信号码列表
  $mobileList = [
      '13111111111',
      '13111111112',
      '13111111113',
      '13111111114',
      '13111111115',
      '13111111116',
      '13111111117',
      '13111111118',
  ];
  // A服务器脚本或使用swoole发送正向的一半
  $serverA = new MessageAsc();
  $iteratorA = $serverA->CreateIterator($mobileList);
  while (!$iteratorA->IsDone()) {
      echo $iteratorA->CurrentItem(), PHP_EOL;
      $iteratorA->Next();
  }
  // B服务器脚本或使用swoole同步发送反向的一半
  $serverB = new MessageDesc();
  $iteratorB = $serverB->CreateIterator($mobileList);
  while (!$iteratorB->IsDone()) {
      echo $iteratorB->CurrentItem(), PHP_EOL;
      $iteratorB->Next();
  }
  说明
 
  完整源码:github.com/zhangyue0503/designpatterns-php/blob/master/06.observer/source/spl_observer.php
 
  彩蛋
 
  PHP中的Iterator接口已经为我们准备好了一套标准的Iterator模式的实现,而且(这里需要画重点),实现这个接口的类可以用foreach来遍历哦!
 
  文档:php.net/manual/zh/class.iterator.php
 
  源码:github.com/zhangyue0503/designpatterns-php/blob/master/07.iterator/source/iterator-php.php
 
  文档中相关的接口都可以看看,更重要的是,PHP的SPL扩展中,也为我们准备了很多常用的迭代器封装。要知道,面试的时候要是能说出这里面的几个来,那面试官可是也会刮目相看的哦!
 

(编辑:我爱资讯网)

【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容!