版权声明:本文出自汪磊的博客,转载请务必注明出处。

一、略模式定义

定义:策略模式定义了一系列的算法,并将每一个算法封装起来,而且使他们之间可以相互替换,策略模式可以在不影响客户端的情况下发生变化。

好了,定义看看就完了,我知道你很烦看定义。

二、策略模式涉及到的三个角色

环境(Context)角色  持有一个Strategy的引用

抽象策略(Strategy)角色  这是一个抽象角色,通常由一个接口或抽象类实现。此角色给出所有的具体策略类所需的接口。

具体策略(ConcreteStrategy)角色  包装了具体的算法或行为。

三、策略模式使用举例

还是赶快看代码吧,知道你看上面定义都烦死了。

我们写一个简单的Demo来对策略模式进行更深的理解,我们模拟游戏中玩家参与一个活动购买金币,普通玩家没有折扣,高级玩家9折优惠,VIP玩家7折优惠。

首先我们定义一个接口,这就是抽象策略角色,如下:

 public interface GameStrategy {

     public void goldCoin();

 }

很简单,就是一个接口。

接下来我们编写具体策略角色类,从需求就能明白,需要有三个具体策略角色类,也就是三个策略对应不同等级玩家。

普通玩家策略类:

 public class normalStrategy implements GameStrategy {

     @Override
public void goldCoin() {
//
System.out.println("普通玩家没有折扣");
}
}

高级玩家策略类:

public class advancedStrategy implements GameStrategy {

    @Override
public void goldCoin() {
// TODO Auto-generated method stub
System.out.println("高级会员9折优惠");
}
}

VIP玩家策略类:

 public class vipStrategy implements GameStrategy {

     @Override
public void goldCoin() {
//
System.out.println("VIP会员7折优惠");
}
}

都很简单,就是简单打印信息。

然后编写环境角色类,如下:

 public class Context {
  //持有一个Strategy的引用
private GameStrategy mGameStrategy; // 构造函数
public Context(GameStrategy strategy) {
this.mGameStrategy = strategy;
} public void setStrategy(GameStrategy strategy) {
this.mGameStrategy = strategy;
} public void goldCoin() {
this.mGameStrategy.goldCoin();
}
}

最后就是调用了,根据不同情况调用不同策略:

 public class main {

     public static void main(String[] args) {
//普通玩家策略
Context context = new Context(new normalStrategy());
context.goldCoin();
//高级玩家策略
context.setStrategy(new advancedStrategy());
context.goldCoin();
//高级玩家策略
context.setStrategy(new vipStrategy());
context.goldCoin();
}
}

不同情况下我们替换为对应策略就可以了。

四、策略模式优缺点

优点

策略模式提供了管理相关的算法族的办法。策略类的等级结构定义了一个算法或行为族。恰当使用继承可以把公共的代码移到父类里面,从而避免代码重复。使用策略模式可以避免使用多重条件(if-else)语句。多重条件语句不易维护,它把采取哪一种算法让子类实现

缺点

通过上面Demo我们会发现调用者必须知道所有的策略类,并自行决定使用哪一个策略类。这就意味着客户端必须理解这些算法的区别,以便适时选择恰当的算法类并且由于策略模式把每个具体的策略实现都单独封装成为类,如果备选的策略很多的话,那么对象的数目就会很可观。

好了上面就是策略模式的介绍了,关键是自己慢慢理解。接下来我们了解一下状态模式。

五、状态模式定义

当一个对象在其内部状态改变的时候改变其行为。这个对象看上去就像是改变了它的类一样。又称状态对象模式,状态模式是对象的行为模式。

简单的理解就是一个类其状态改变了,那么其功能也相应的改变了。比如水变为冰,状态改变了,其功能也发生了相应变化。

六、状态模式角色

 Context(环境类):环境类又称为上下文类,它是拥有多种状态的对象。由于环境类的状态存在多样性且在不同状态下对象的行为有所不同,因此将状态独立出去形成单独的状态类。在环境类中维护一个抽象状态类State的实例,这个实例定义当前状态,在具体实现时,它是一个State子类的对象。

 State(抽象状态类):它用于定义一个接口以封装与环境类的一个特定状态相关的行为,在抽象状态类中声明了各种不同状态对应的方法,而在其子类中实现类这些方法,由于不同状态下对象的行为可能不同,因此在不同子类中方法的实现可能存在不同,相同的方法可以写在抽象状态类中。

 ConcreteState(具体状态类):它是抽象状态类的子类,每一个子类实现一个与环境类的一个状态相关的行为,每一个具体状态类对应环境的一个具体状态,不同的具体状态类其行为有所不同。

七、状态模式使用举例

接下来我们编写一个小Demo来进行更深入理解:在办公系统中,一个文件会有不同状态:未处理,正在处理,已经处理。我们就简单模拟一下文件状态的切换。

首先我们看下抽象状态类,很简单就是定义一个接口:

 public interface State {

     void fileState();
}

接下来看下三个具体状态类:

未处理状态类

 public class NotDealState  implements State{

     @Override
public void fileState() {
// TODO Auto-generated method stub
System.out.println("文件未处理状态");
}
}

正在处理状态类:

 public class DealingState implements State{

     @Override
public void fileState() {
// TODO Auto-generated method stub
System.out.println("文件正在处理状态");
}
}

已经处理状态类

 public class HasDealState implements State{

     @Override
public void fileState() {
// TODO Auto-generated method stub
System.out.println("文件已经处理状态");
}
}

是不是很简单,同样也只是打印一些简单的信息罢了。

然后看下上下文类:

 public class FileContext {

     //默认情况下未处理状态
private State fileState = new NotDealState(); /**
* 文件状态切换为未处理状态
*/
public void notDeal(){ this.fileState = new NotDealState();
} /**
* 文件状态切换为正在处理状态
*/
public void Dealing(){ this.fileState = new DealingState();
} /**
* 文件状态切换为已经处理状态
*/
public void HasDeal(){ this.fileState = new HasDealState();
} /**
* 获取当前文件的状态
*/
public void getFileState(){
fileState.fileState();
}
}

上下文类主要就是封装当前文件状态并且对外提供状态切换的方法

最后看下怎么使用吧:

 public class Client {

     public static void main(String[] args) {

         FileContext fileContext = new FileContext();
fileContext.getFileState();
//切换为正在处理状态
fileContext.Dealing();
fileContext.getFileState();
//切换为已经处理状态
fileContext.HasDeal();
fileContext.getFileState();
}
}

外部使用调用相应切换状态方法就可以了,很简单,没有什么需要特别说明的。

八、状态模式与策略模式区别以及联系

策略模式与状态模式及其相似,但是二者有其内在的差别,策略模式将具体策略类暴露出去,调用者需要具体明白每个策略的不同之处以便正确使用。而状态模式状态的改变是由其内部条件来改变的,与外界无关,二者在思想上有本质区别。

好了,状态模式与策略模式讲解到此为止,二者区别联系不是几句话就能说清楚,关键是思想上的不同,具体感悟需要在实践中自己慢慢体会。

Java设计模式之策略模式与状态模式的更多相关文章

  1. Java进阶篇设计模式之十二 ---- 备忘录模式和状态模式

    前言 在上一篇中我们学习了行为型模式的策略模式(Strategy Pattern)和模板模式(Template Pattern).本篇则来学习下行为型模式的两个模式,备忘录模式(Memento Pat ...

  2. 【设计模式】 模式PK:策略模式VS状态模式

    1.概述 行为类设计模式中,状态模式和策略模式是亲兄弟,两者非常相似,我们先看看两者的通用类图,把两者放在一起比较一下. 策略模式(左)和状态模式(右)的通用类图. 两个类图非常相似,都是通过Cont ...

  3. 设计模式之第21章-状态模式(Java实现)

    设计模式之第21章-状态模式(Java实现) “what are you 干啥了?怎么这么萎靡不振?”“昨晚又是补新番,又是补小笼包,睡得有点晚啊.话说杨过的那个雕兄真是太好了,每天给找蛇胆,又陪练武 ...

  4. 设计模式之策略模式和状态模式(strategy pattern & state pattern)

    本文来讲解一下两个结构比较相似的行为设计模式:策略模式和状态模式.两者单独的理解和学习都是比较直观简单的,但是实际使用的时候却并不好实践,算是易学难用的设计模式吧.这也是把两者放在一起介绍的原因,经过 ...

  5. Java 策略模式和状态模式

    本文是转载的,转载地址:大白话解释Strategy模式和State模式的区别 先上图: 本质上讲,策略模式和状态模式做得是同一件事:去耦合.怎么去耦合?就是把干什么(语境类)和怎么干(策略接口)分开, ...

  6. java - 策略模式、状态模式、卫语句,避免多重if-else(转)

    前言 当代码中出现多重if-else语句或者switch语句时.弊端之一:如果这样的代码出现在多处,那么一旦出现需求变更,就需要把所有地方的if-else或者switch代码进行更改,要是遗漏了某一处 ...

  7. Java设计模式之十二 ---- 备忘录模式和状态模式

    前言 在上一篇中我们学习了行为型模式的策略模式(Strategy Pattern)和模板模式(Template Pattern).本篇则来学习下行为型模式的两个模式,备忘录模式(Memento Pat ...

  8. Java重构-策略模式、状态模式、卫语句

    前言 当代码中出现多重if-else语句或者switch语句时.弊端之一:如果这样的代码出现在多处,那么一旦出现需求变更,就需要把所有地方的if-else或者switch代码进行更改,要是遗漏了某一处 ...

  9. 【转】Java重构-策略模式、状态模式、卫语句

    前言 当代码中出现多重if-else语句或者switch语句时.弊端之一:如果这样的代码出现在多处,那么一旦出现需求变更,就需要把所有地方的if-else或者switch代码进行更改,要是遗漏了某一处 ...

随机推荐

  1. Codeforces Round #356 (Div. 1) C. Bear and Square Grid

    C. Bear and Square Grid time limit per test 3 seconds memory limit per test 256 megabytes input stan ...

  2. React Native学习(四)—— 写一个公用组件(头部)

    本文基于React Native 0.52 Demo上传到Git了,有需要可以看看,写了新内容会上传的.Git地址 https://github.com/gingerJY/React-Native-D ...

  3. Thinking in Java学习笔记-泛型和类型安全的容器

    示例: public class Apple { private static long counter; private final long id = counter++; public long ...

  4. 对于hive使用的一点记录

    最近一段时间因工作需要接触了一些hive上的使用!当然大部分都是比较基本的使用,仅当入门!各位看到有不足之处望多多指正! 废话不多说,开始: 首先是创建数据库 create database '数据库 ...

  5. Servlet面试题归纳

    尊重原创:http://blog.csdn.net/caohaicheng/article/details/38116481 1.说一说Servlet生命周期 Servlet生命周期包括三部分: 初始 ...

  6. Unity Instantiate各函数执行顺序

    GameObject go = Instantiate(aaa) ;        go.GetComponent<NewBehaviourScript>().Init(); Instan ...

  7. Spark_总结一

    Spark_总结一 1.Spark介绍     1.1什么是Spark?     Apache Spark是一个开源的集群计算框架,使数据计算更快(高效运行,快速开发)          1.2Spa ...

  8. Logback日志配置的简单使用

    Logback介绍 Logback是由log4j创始人设计的又一个开源日志组件.logback当前分成三个模块:logback-core,logback- classic和logback-access ...

  9. 使用pscp命令在Windows与linux之间拷贝文件

    在Linux环境下做开发时,我们可能会在windows下远程控制Linux系统,本文介绍如何使用putty的pscp.exe在Windows和 Linux(本文使用fedora)之间copy文件. 工 ...

  10. ThinkPhp_5框架开发【整理】

    ================================================== ThinkPhp_5心得笔记 ---------------------------------- ...