PHP设计模式入门之状态模式原理与实现方法分析


Posted in PHP onApril 26, 2020

本文实例讲述了PHP设计模式入门之状态模式原理与实现方法。分享给大家供大家参考,具体如下:

想必大家都用过自动售卖的自动饮料机吧,塞入硬币或纸币,选择想要的饮料,饮料就会在机器的下方滚出。大家有没有相关如果用程序去写一个饮料机要怎么样实现呢?

首先我们可以分享一下这部饮料机有几种状态

一、没有钱的状态

二、有钱的状态

三、售出的状态

四、销售一空的状态

好吧,知道了这些状态之后我们开始写代码了!

JuiceMachine.php

<?php
/**
 * 饮料机
 * @author ben
 *
 */
class JuiceMachine{
 /**
 * 糖果机一共存在四种状态:没钱,有钱,成功售出以及销售一空
 * 
 * 没钱的状态
 * @var INT
 */
 const NOMONEY = 0;
 
 /**
 * 有钱的状态
 * @var INT
 */
 const HASMONEY = 1;
 
 /**
 * 成功售出的状态
 * @var INT
 */
 const SOLD = 2;
 
 /**
 * 销售一空的状态
 * @var INT
 */
 const SOLDOUT = 3;
 
 /**
 * 记录糖果机当前的状态,初始化状态为售空
 * @var INT
 */
 private $_state = JuiceMachine::SOLDOUT;
 
 /**
 * 该变量用于记录饮料机中饮料的数量
 */
 private $_count; 
 
 /**
 * 构造方法,最主要是用来初始化count和state属性的
 */
 public function __construct($count){
   $this->_count = $count;
   //当饮料机中的饮料数量大于零时,将饮料机的状态重置为没有钱的状态。
   if($this->_count > 0){
     $this->_state = JuiceMachine::NOMONEY;
   }
 }
 
 /**
 * 投入硬币
 */
 public function insertCoin(){
   if($this->_state == JuiceMachine::HASMONEY ){
     echo "you can't insert another coin!<br />";
   }elseif($this->_state == JuiceMachine::NOMONEY){
     echo "you just insert a coin<br />";
     $this->_state = JuiceMachine::HASMONEY;
   }elseif($this->_state == JuiceMachine::SOLD){
     echo "wait a minute, we are giving you a bottle of juice<br />";
   }elseif($this->_state == JuiceMachine::SOLDOUT){
     echo "you can't insert coin, the machine is already soldout<br />";
   }
 }
 
 /**
 * 退回硬币
 */
 public function retreatCoin(){
   if($this->_state == JuiceMachine::HASMONEY ){
     echo "coin return!<br />";
     $this->_state = JuiceMachine::NOMONEY;
   }elseif($this->_state == JuiceMachine::NOMONEY){
     echo "you have'nt inserted a coin yet<br />";
   }elseif($this->_state == JuiceMachine::SOLD){
     echo "sorry, you already clicked the botton<br />";
   }elseif($this->_state == JuiceMachine::SOLDOUT){
     echo "you have'nt inserted a coin yet<br />";
   }
 }
 
 /**
 * 点击饮料对应的按钮
 */
 public function clickButton(){
   if($this->_state == JuiceMachine::HASMONEY ){
     echo "you clicked, we are giving you a bottle of juice...<br />";
     $this->_state = JuiceMachine::SOLD;  //改变饮料机的状态为售出模式
     $this->dispend();
   }elseif($this->_state == JuiceMachine::NOMONEY){
     echo "you clicked,but you hav'nt inserted a coin yet<br />";
   }elseif($this->_state == JuiceMachine::SOLD){
     echo "click twice does'nt get you two bottle of juice<br />";
   }elseif($this->_state == JuiceMachine::SOLDOUT){
     echo "you clicked, but the machine is already soldout<br />";
   }
 }
 
 /**
 * 发放饮料
 */
 public function dispend(){
   if($this->_state == JuiceMachine::HASMONEY ){
     echo "please click the button first<br />";
   }elseif($this->_state == JuiceMachine::NOMONEY){
     echo "you need to pay first<br />";
   }elseif($this->_state == JuiceMachine::SOLD){
     echo "now you get you juice<br />";
     //饮料机中的饮料数量减一
     $this->_count--;
     if($this->_count <= 0){
       echo "opps, runing out of juice<br />";
       //如果这时饮料机中没有饮料了,将饮料机的状态重置为销售一空
       $this->_state = JuiceMachine::SOLDOUT;
     }else{
       //将饮料机的状态重置为没有钱
       $this->_state = JuiceMachine::NOMONEY;
     }
   }elseif($this->_state == JuiceMachine::SOLDOUT){
     //其实这种情况不应该出现
     echo "opps, it appears that we don't have any juice left<br />";
   }
 }
}

index.php

<?php
require_once 'JuiceMachine.php';
 
$juiceMachine = new JuiceMachine(1);
 
$juiceMachine->insertCoin();
$juiceMachine->clickButton();

运行的结果是:

you just insert a coin
you clicked, we are giving you a bottle of juice...
now you get you juice
opps, runing out of juice

到目前为止我们的程序运行良好,没有出现什么问题,但是从这些多重的if判断中你是否嗅到了坏代码的味道呢?有一天问题终于出现了,老板希望当用户点击按钮时有10%的概率拿到两瓶饮料,我们需要为饮料机多加一个状态,这时去修改代码就成为了一种灾难,而且很可能会影响到之前的代码,带来新的bug,看看状态模式如何帮助我们度过难关吧!

状态模式的官方定义是:状态模式允许对象在内部状态改变是改变它的行为,对象看起来好像是修改了它的类

用uml类图表示如下:

PHP设计模式入门之状态模式原理与实现方法分析

在我们这个项目中的实际类图如下:

PHP设计模式入门之状态模式原理与实现方法分析

具体实现代码:

State.php

<?php
interface State{
  
  /**
   * 插入硬币
   */
  public function insertCoin();
  
  /**
   * 回退硬币
   */
  public function retreatCoin();
  
  /**
   * 点击按钮
   */
  public function clickButton();
  
  /**
   * 发放饮料
   */
  public function dispend();
}

NomoneyState.php

<?php
require_once 'State.php';
class NomoneyState implements State{
  
  /**
   * 饮料机的实例
   * 
   * @var object
   */
  private $_juiceMachine;
  
  /**
   * 构造方法,主要用于初始化饮料机实例
   * 
   */
  public function __construct($juiceMachine){
    $this->_juiceMachine = $juiceMachine;
  }
  
 /* (non-PHPdoc)
   * @see State::insertCoin()
   */
  public function insertCoin()
  {
    // TODO Auto-generated method stub
    echo "you just insert a coin<br />";
    //将饮料机的状态切换成有钱的状态
    $this->_juiceMachine->setState($this->_juiceMachine->getHasmoneyState());
  }
 
 /* (non-PHPdoc)
   * @see State::retreatCoin()
   */
  public function retreatCoin()
  {
    // TODO Auto-generated method stub
    echo "you have'nt inserted a coin yet<br />";
  }
 
 /* (non-PHPdoc)
   * @see State::clickButton()
   */
  public function clickButton()
  {
    // TODO Auto-generated method stub
    echo "you clicked,but you hav'nt inserted a coin yet<br />";
  }
 
 /* (non-PHPdoc)
   * @see State::dispend()
   */
  public function dispend()
  {
    // TODO Auto-generated method stub
    echo "you need to pay first<br />";
  }
}

HasmoneyState.php

<?php
require_once 'State.php';
 
class HasmoneyState implements State
{
 
  /**
   * 饮料机的实例
   *
   * @var object
   */
  private $_juiceMachine;
 
  /**
   * 构造方法,主要用于初始化饮料机实例
   */
  public function __construct($juiceMachine)
  {
    $this->_juiceMachine = $juiceMachine;
  }
  
  /*
   * (non-PHPdoc) @see State::insertCoin()
   */
  public function insertCoin()
  {
    // TODO Auto-generated method stub
    echo "you can't insert another coin!<br />";
  }
  
  /*
   * (non-PHPdoc) @see State::retreatCoin()
   */
  public function retreatCoin()
  {
    // TODO Auto-generated method stub
    echo "coin return!<br />";
    $this->_juiceMachine->setState($this->_juiceMachine->getNomoneyState());
  }
  
  /*
   * (non-PHPdoc) @see State::clickButton()
   */
  public function clickButton()
  {
    // TODO Auto-generated method stub
    echo "you clicked, we are giving you a bottle of juice...<br />";
    // 改变饮料机的状态为售出模式
    $rand = mt_rand(0, 0);
    // 当随机数为0(即1/10的概率)并且饮料机中还有1瓶以上的饮料时
    if ($rand == 0 && $this->_juiceMachine->getCount() > 1) {
      $this->_juiceMachine->setState($this->_juiceMachine->getWinnerState());
    } else {
      $this->_juiceMachine->setState($this->_juiceMachine->getSoldState());
    }
  }
  
  /*
   * (non-PHPdoc) @see State::dispend()
   */
  public function dispend()
  {
    // TODO Auto-generated method stub
    echo "please click the button first<br />";
  }
}

SoldoutState.php

<?php
require_once 'State.php';
class SoldoutState implements State{
  
  /**
   * 饮料机的实例
   *
   * @var object
   */
  private $_juiceMachine;
  
  /**
   * 构造方法,主要用于初始化饮料机实例
   *
   */
  public function __construct($juiceMachine){
    $this->_juiceMachine = $juiceMachine;
  }
  
 /* (non-PHPdoc)
   * @see State::insertCoin()
   */
  public function insertCoin()
  {
    // TODO Auto-generated method stub
    echo "you can't insert coin, the machine is already soldout<br />";
  }
 
 /* (non-PHPdoc)
   * @see State::retreatCoin()
   */
  public function retreatCoin()
  {
    // TODO Auto-generated method stub
    echo "you have'nt inserted a coin yet<br />";
  }
 
 /* (non-PHPdoc)
   * @see State::clickButton()
   */
  public function clickButton()
  {
    // TODO Auto-generated method stub
    echo "you clicked, but the machine is already soldout<br />";
  }
 
 /* (non-PHPdoc)
   * @see State::dispend()
   */
  public function dispend()
  {
    // TODO Auto-generated method stub
    echo "opps, it appears that we don't have any juice left<br />";
  }
}

SoldState.php

<?php
require_once 'State.php';
class SoldState implements State{
  
  /**
   * 饮料机的实例
   *
   * @var object
   */
  private $_juiceMachine;
  
  /**
   * 构造方法,主要用于初始化饮料机实例
   *
   */
  public function __construct($juiceMachine){
    $this->_juiceMachine = $juiceMachine;
  }
  
 /* (non-PHPdoc)
   * @see State::insertCoin()
   */
  public function insertCoin()
  {
    // TODO Auto-generated method stub
    echo "wait a minute, we are giving you a bottle of juice<br />";
  }
 
 /* (non-PHPdoc)
   * @see State::retreatCoin()
   */
  public function retreatCoin()
  {
    // TODO Auto-generated method stub
    echo "sorry, you already clicked the botton<br />";
  }
 
 /* (non-PHPdoc)
   * @see State::clickButton()
   */
  public function clickButton()
  {
    // TODO Auto-generated method stub
    echo "click twice does'nt get you two bottle of juice<br />";
  }
 
 /* (non-PHPdoc)
   * @see State::dispend()
   */
  public function dispend()
  {
    $this->_juiceMachine->decJuice();
    if($this->_juiceMachine->getCount() <= 0){
      echo "opps, runing out of juice<br />";
      //如果这时饮料机中没有饮料了,将饮料机的状态重置为销售一空
       $this->_juiceMachine->setState($this->_juiceMachine->getSoldoutState());
    }else{
      //将饮料机的状态重置为没有钱
       $this->_juiceMachine->setState($this->_juiceMachine->getNomoneyState());
    }
  }
  
}

WinnerState.php

<?php
require_once 'State.php';
 
class WinnerState implements State
{
 
  /**
   * 饮料机的实例
   *
   * @var object
   */
  private $_juiceMachine;
 
  /**
   * 构造方法,主要用于初始化饮料机实例
   */
  public function __construct($juiceMachine)
  {
    $this->_juiceMachine = $juiceMachine;
  }
  
  /*
   * (non-PHPdoc) @see State::insertCoin()
   */
  public function insertCoin()
  {
    // TODO Auto-generated method stub
    echo "wait a minute, we are giving you a bottle of juice<br />";
  }
  
  /*
   * (non-PHPdoc) @see State::retreatCoin()
   */
  public function retreatCoin()
  {
    // TODO Auto-generated method stub
    echo "sorry, you already clicked the botton<br />";
  }
  
  /*
   * (non-PHPdoc) @see State::clickButton()
   */
  public function clickButton()
  {
    // TODO Auto-generated method stub
    echo "click twice does'nt get you two bottle of juice<br />";
  }
  
  /*
   * (non-PHPdoc) @see State::dispend()
   */
  public function dispend()
  {
    echo "you are a winner! you get two bottle of juice!<br />";
    $this->_juiceMachine->decJuice();
    if ($this->_juiceMachine->getCount() > 0) {
      $this->_juiceMachine->decJuice();
      if ($this->_juiceMachine->getCount() <= 0) {
        echo "opps, runing out of juice<br />";
        // 如果这时饮料机中没有饮料了,将饮料机的状态重置为销售一空
        $this->_juiceMachine->setState($this->_juiceMachine->getSoldoutState());
      } else {
        // 将饮料机的状态重置为没有钱
        $this->_juiceMachine->setState($this->_juiceMachine->getSoldoutState());
      }
    } else {
      echo "opps, runing out of juice<br />";
      // 如果这时饮料机中没有饮料了,将饮料机的状态重置为销售一空
      $this->_juiceMachine->setState($this->_juiceMachine->getSoldoutState());
    }
  }
}

JuiceMachine.php

<?php
require_once './state/NomoneyState.php';
require_once './state/HasmoneyState.php';
require_once './state/SoldState.php';
require_once './state/SoldoutState.php';
require_once './state/WinnerState.php';
 
class JuiceMachine
{
 
  /**
   * 记录糖果机当前的状态,初始化状态为售空
   * 
   * @var object
   */
  private $_state;
 
  /**
   * 该变量用于记录饮料机中饮料的数量
   */
  private $_count;
 
  /**
   * 构造方法,最主要是用来初始化count和state属性的
   */
  public function __construct($count)
  {
    $this->_state = new SoldoutState($this);
    $this->_count = $count;
    // 当饮料机中的饮料数量大于零时,将饮料机的状态重置为没有钱的状态。
    if ($this->_count > 0) {
      $this->_state = new NomoneyState($this);
    }
  }
  
  /*
   * (non-PHPdoc) @see State::insertCoin()
   */
  public function insertCoin()
  {
    // TODO Auto-generated method stub
    $this->_state->insertCoin();
  }
  
  /*
   * (non-PHPdoc) @see State::retreatCoin()
   */
  public function retreatCoin()
  {
    // TODO Auto-generated method stub
    $this->_state->retreatCoin();
  }
  
  /*
   * (non-PHPdoc) @see State::clickButton()
   */
  public function clickButton()
  {
    $this->_state->clickButton();
    //其实发放糖果是在用户点击完按钮后机器内部进行的所有没有必要再写一个dispend方法
    $this->_state->dispend();
  }
  
  /**
   * 设置糖果机的状态
   * 
   * @param State $state
   */
  public function setState(State $state)
  {
    $this->_state = $state;
  }
  
  /**
   * 获取没有钱的状态
   */
  public function getNomoneyState(){
    return new NomoneyState($this);
  }
  
  /**
   * 获取有钱的状态
   */
  public function getHasmoneyState(){
    return new HasmoneyState($this);
  }
  
  /**
   * 获取售出的状态
   */
  public function getSoldState(){
    return new SoldState($this);
  }
  
  /**
   * 获取销售一空的状态
   */
  public function getSoldoutState(){
    return new SoldoutState($this);
  }
  
  /**
   * 获取幸运者的状态
   */
  public function getWinnerState(){
    return new WinnerState($this);
  }
  
  /**
   * 获取饮料机中饮料的数量
   */
  public function getCount(){
    return $this->_count;
  }
  
  /**
   * 将饮料数量减一
   */
  public function decJuice(){
    echo "now you get you juice<br />";
    //饮料机中的饮料数量减一
    $this->_count--;
  }
  
}

index.php

<?php
require_once 'JuiceMachine.php';
 
$juiceMachine = new JuiceMachine(2);
 
$juiceMachine->insertCoin();
$juiceMachine->clickButton();

希望本文所述对大家PHP程序设计有所帮助。

PHP 相关文章推荐
刚才在简化php的库,结果发现很多东西
Dec 31 PHP
浅谈web上存漏洞及原理分析、防范方法(文件名检测漏洞)
Jun 29 PHP
phpQuery占用内存过多的处理方法
Nov 13 PHP
php获取根域名方法汇总
Oct 28 PHP
PHP生成器简单实例
May 13 PHP
PHP处理postfix邮件内容的方法
Jun 16 PHP
Linux安装配置php环境的方法
Jan 14 PHP
Zend Framework教程之MVC框架的Controller用法分析
Mar 07 PHP
ThinkPHP中order()使用方法详解
Apr 19 PHP
php实现xml转换数组的方法示例
Feb 03 PHP
Laravel框架实现的使用smtp发送邮件功能示例
Mar 12 PHP
php转换上传word文件为PDF的方法【基于COM组件】
Jun 10 PHP
PHP设计模式入门之迭代器模式原理与实现方法分析
Apr 26 #PHP
PHP中迭代器的简单实现及Yii框架中的迭代器实现方法示例
Apr 26 #PHP
PHP设计模式之迭代器模式Iterator实例分析【对象行为型】
Apr 26 #PHP
Yii Framework框架开发微信公众平台示例
Apr 26 #PHP
PHP随机生成中文段落示例【测试网站内容时使用】
Apr 26 #PHP
PHP过滤器 filter_has_var() 函数用法实例分析
Apr 23 #PHP
PHP优化之批量操作MySQL实例分析
Apr 23 #PHP
You might like
php实现执行某一操作时弹出确认、取消对话框
2013/12/30 PHP
php中$美元符号与Zen Coding冲突问题解决方法分享
2014/05/28 PHP
PHP+Redis事务解决高并发下商品超卖问题(推荐)
2020/08/03 PHP
在JavaScript中遭遇级联表达式陷阱
2007/03/08 Javascript
js借助ActiveXObject实现创建文件
2013/09/29 Javascript
JavaScript模板引擎用法实例
2015/07/10 Javascript
JS简单限制textarea内输入字符数量的方法
2015/10/14 Javascript
基于Jquery代码实现支持PC端手机端幻灯片代码
2015/11/17 Javascript
不能不知道的10个angularjs英文学习网站
2016/03/23 Javascript
JS转换HTML转义符的方法
2016/08/24 Javascript
简单的js表格操作
2016/09/24 Javascript
jQuery旋转插件jqueryrotate用法详解
2016/10/13 Javascript
jQuery实现优雅的弹窗效果(6)
2017/02/08 Javascript
详解vue嵌套路由-query传递参数
2017/05/23 Javascript
详解vue.js移动端配置flexible.js及注意事项
2019/04/10 Javascript
vue+koa2实现session、token登陆状态验证的示例
2019/08/30 Javascript
angular inputNumber指令输入框只能输入数字的实现
2019/12/03 Javascript
Vue+Java+Base64实现条码解析的示例
2020/09/23 Javascript
微信小程序自定义底部弹出框动画
2020/11/18 Javascript
在Python的Bottle框架中使用微信API的示例
2015/04/23 Python
python编写爬虫小程序
2015/05/14 Python
Python断言assert的用法代码解析
2018/02/03 Python
python判断设备是否联网的方法
2018/06/29 Python
Python利用ORM控制MongoDB(MongoEngine)的步骤全纪录
2018/09/13 Python
对python GUI实现完美进度条的示例详解
2018/12/13 Python
selenium+超级鹰实现模拟登录12306
2021/01/24 Python
html5中去掉input type date默认样式的方法
2018/09/06 HTML / CSS
MCM英国官网:奢侈皮具制品
2017/04/18 全球购物
房屋改造计划书
2014/01/10 职场文书
市场安全管理制度
2014/01/26 职场文书
就业协议书范本
2014/04/11 职场文书
我为党旗添光彩演讲稿
2014/09/10 职场文书
大型主题婚礼活动策划方案
2014/09/15 职场文书
电子商务实训报告总结
2014/11/05 职场文书
初三英语教学计划
2015/01/23 职场文书
2015年企业工作总结范文
2015/04/28 职场文书