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+mysql留言本源码
Nov 11 PHP
PHP操作MongoDB时的整数问题及对策说明
May 02 PHP
PHP移动文件指针ftell()、fseek()、rewind()函数总结
Nov 18 PHP
thinkphp中session和cookie无效的解决方法
Dec 19 PHP
mysql_connect localhost和127.0.0.1的区别(网络层阐述)
Mar 26 PHP
PHP微信开发之查询城市天气
Jun 23 PHP
PHP使用递归算法无限遍历数组示例
Jan 13 PHP
完美解决thinkphp唯一索引重复时出错的问题
Mar 31 PHP
PHP基于简单递归函数求一个数阶乘的方法示例
Apr 26 PHP
作为PHP程序员你要知道的另外一种日志
Jul 30 PHP
Django 标签筛选的实现代码(一对多、多对多)
Sep 05 PHP
PHP连接MySQL数据库操作代码实例解析
Jul 11 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中的global
2014/08/19 PHP
PHP中$_SERVER使用说明
2015/07/05 PHP
thinkPHP5框架闭包函数与子查询传参用法示例
2018/08/02 PHP
经典的带阴影的可拖动的浮动层
2006/06/26 Javascript
定义select的边框颜色
2008/04/28 Javascript
jQuery 表单验证扩展代码(二)
2010/10/20 Javascript
jquery键盘事件介绍
2011/01/31 Javascript
angularjs 处理多个异步请求方法汇总
2015/01/06 Javascript
freemarker判断对象是否为空的方法
2015/08/13 Javascript
JavaScript提升性能的常用技巧总结【经典】
2016/06/20 Javascript
jQuery webuploader分片上传大文件
2016/11/07 Javascript
JavaScript反弹动画效果的实现代码
2017/07/13 Javascript
javaScript和jQuery自动加载简单代码实现方法
2017/11/24 jQuery
vue使用 better-scroll的参数和方法详解
2018/01/25 Javascript
一个因@click.stop引发的bug的解决
2019/01/08 Javascript
使用原生JS实现火锅点餐小程序(面向对象思想)
2019/12/10 Javascript
[01:46]辉夜杯—打造中国DOTA新格局
2015/12/25 DOTA
Python AES加密模块用法分析
2017/05/22 Python
Python绘制3d螺旋曲线图实例代码
2017/12/20 Python
一个月入门Python爬虫学习,轻松爬取大规模数据
2018/01/03 Python
pytorch: tensor类型的构建与相互转换实例
2018/07/26 Python
python实现写数字文件名的递增保存文件方法
2018/10/25 Python
python之线程通过信号pyqtSignal刷新ui的方法
2019/01/11 Python
Python异常处理例题整理
2019/07/07 Python
matplotlib.pyplot.matshow 矩阵可视化实例
2020/06/16 Python
CSS3制作hover下划线动画
2017/03/27 HTML / CSS
HTML5中的网络存储实现方式
2020/04/28 HTML / CSS
西班牙创意礼品和小工具网上商店:Curiosite
2016/07/26 全球购物
苏宁红孩子母婴商城:redbaby
2017/02/12 全球购物
运动鞋、街头服装、手表和手袋的实时市场:StockX
2020/11/25 全球购物
物流专业大学生职业生涯规划书范文
2014/01/15 职场文书
仓库规划计划书
2014/04/28 职场文书
安全生产标语大全
2014/10/06 职场文书
团代会开幕词
2015/01/28 职场文书
2015年敬老院工作总结
2015/05/18 职场文书
Oracle中update和select 关联操作
2022/01/18 Oracle