java中的策略模式(策略模式java)

前言当代码中出现多重if-else语句或者switch语句时。弊端之一:如果这样的代码出现在多处,那么一旦出现需求变更,就需要把所有地方的if-else或者switch代码进行更改,要是遗漏了某一处,那么程序就会出错。弊端之二:代码逻辑难以理解。卫语句卫语句的使用,并不能改变前言说的弊端:一旦需求更改,需要修改所有使用更改需求的if-else代码块。不过使用卫语句可以让自己或者代码…

大家好,又见面了,我是你们的朋友全栈君。

前言

当代码中出现多重if-else语句或者switch语句时。弊端之一:如果这样的代码出现在多处,那么一旦出现需求变更,就需要把所有地方的if-else或者switch代码进行更改,要是遗漏了某一处,那么程序就会出错。弊端之二:代码逻辑难以理解。

卫语句

卫语句的使用,并不能改变前言说的弊端:一旦需求更改,需要修改所有使用更改需求的if-else代码块。不过使用卫语句可以让自己或者代码维护人员很容易的了解代码的含义。

代替异常分支

所谓的异常分支就是普通情况之外的分支,这种情况的下的if-else可以用

if (condition)
    return obj;

代替。

代替多重if-else分支

这个情况的代替方式是可以为

if (isSunshine()) {
    // 晴天时处理逻辑
}
if (isRain()) {
    // 下雨时处理逻辑
}
if (isOvercast()) {
    // 阴天时处理逻辑
}
...

策略模式

使用策略模式可以代替多重if-elseswitch语句,让代码维护变得更加简单。
策略模式UML:
这里写图片描述

  • 环境(Context)角色:持有一个Strategy的引用
  • 抽象策略(Strategy)角色:这是一个抽象角色,通常由一个接口或抽象类实现
  • 具体策略(ConcreteStrategy)角色:包装了相关的算法或行为

策略模式代码模板:

package xyz.zeling.test.strategy.template;

import xyz.zeling.test.strategy.template.base.Strategy;

/** * @description 环境角色 * @author zeling * @date 2018年1月14日 下午8:43:58 */
public class Context { 
   

    /** * 策略对象 */
    private Strategy strategy;

    /** * @param strategy 具体策略对象 */
    public Context(Strategy strategy) {
        this.strategy = strategy;
    }

    /** * @description 执行策略方法 * @date 2018年1月14日 下午8:43:31 */
    public void contextInterface() {
        strategy.strategyInterface();
    }
}
package xyz.zeling.test.strategy.template.base;

/** * @description 抽象策略角色 * @author zeling * @date 2018年1月14日 下午8:41:14 */
public interface Strategy { 
   

    /** * @description 策略方法 * @date 2018年1月14日 下午8:41:00 */
    void strategyInterface();
}
package xyz.zeling.test.strategy.template;

import xyz.zeling.test.strategy.template.base.Strategy;

/** * @description 具体策略类A * @author zeling * @date 2018年1月14日 下午8:45:00 */
public class ConcreteStrategyA implements Strategy { 
   

    @Override
    public void strategyInterface() {
        // TODO Auto-generated method stub
    }
}
package xyz.zeling.test.strategy.template;

import xyz.zeling.test.strategy.template.base.Strategy;

/** * @description 具体策略类B * @author zeling * @date 2018年1月14日 下午8:45:00 */
public class ConcreteStrategyB implements Strategy { 
   

    @Override
    public void strategyInterface() {
        // TODO Auto-generated method stub
    }
}
package xyz.zeling.test.strategy.template;

import xyz.zeling.test.strategy.template.base.Strategy;

/** * @description 具体策略类C * @author zeling * @date 2018年1月14日 下午8:45:00 */
public class ConcreteStrategyC implements Strategy { 
   

    @Override
    public void strategyInterface() {
        // TODO Auto-generated method stub
    }
}

实例应用:水果有不同种类,每种水果对应不同价格,用策略模式实现。

package xyz.zeling.test.strategy.demo.base;

/** * @description 抽象策略,水果接口 * @author zeling * @date 2018年1月14日 下午3:01:20 */
public interface Fruit { 
   
    /** * @description 输出对应价格 * @date 2018年1月14日 下午3:03:07 */
    void price();
}
package xyz.zeling.test.strategy.demo;

import xyz.zeling.test.strategy.demo.base.Fruit;

/** * @description 具体策略,苹果类 * @author zeling * @date 2018年1月14日 下午3:04:17 */
public class Apple implements Fruit { 
   

    @Override
    public void price() {
        System.out.println("苹果的价格!");
    }
}
package xyz.zeling.test.strategy.demo;

import xyz.zeling.test.strategy.demo.base.Fruit;

/** * @description 具体策略,香蕉 * @author zeling * @date 2018年1月14日 下午3:06:52 */
public class Banana implements Fruit { 
   

    @Override
    public void price() {
        System.out.println("香蕉的价格!");
    }
}
package xyz.zeling.test.strategy.demo;

import xyz.zeling.test.strategy.demo.base.Fruit;

/** * @description 具体策略,梨 * @author zeling * @date 2018年1月14日 下午3:08:19 */
public class Pear implements Fruit { 
   

    @Override
    public void price() {
        System.out.println("梨的价格!");
    }
}
package xyz.zeling.test.strategy.demo;

import xyz.zeling.test.strategy.demo.base.Fruit;

/** * @description 策略环境,输出水果价格 * @author zeling * @date 2018年1月14日 下午3:09:31 */
public class FruitPrice { 
   

    /** * 策略对象 */
    private Fruit fruit;

    /** * @param fruit 策略对象 */
    public FruitPrice(Fruit fruit) {
        this.fruit = fruit;
    }

    /** * @description 输出水果价格 * @date 2018年1月14日 下午3:12:26 */
    public void printFruitPrice() {
        fruit.price();
    }
}

客户端

    /** * @description 使用策略模式:针对一组算法,将每一个算法封装到具有共同接口的独立的类 * @date 2018年1月14日 下午3:17:11 */
    public static void useStrategy() {
        // 具体使用策略
        Fruit apple = new Apple();
        // 将策略放入环境中并执行策略
        new FruitPrice(apple).printFruitPrice();
    }

状态模式

状态模式类图:
这里写图片描述
说明:

  • 环境(Context)角色,也成上下文:定义客户端所感兴趣的接口,并且保留一个具体状态类的实例。这个具体状态类的实例给出此环境对象的现有状态
  • 抽象状态(State)角色:定义一个接口,用以封装环境(Context)对象的一个特定的状态所对应的行为
  • 具体状态(ConcreteState)角色:每一个具体状态类都实现了环境(Context)的一个状态所对应的行为

策略模式代码模板:

package xyz.zeling.test.state.template.base;

/** * @description 抽象状态角色 * @author zeling * @date 2018年1月14日 下午8:41:14 */
public interface State { 
   

    /** * @description 处理方法 * @date 2018年1月14日 下午8:41:00 */
    void handle();
}
package xyz.zeling.test.state.template;

import xyz.zeling.test.state.template.base.State;

/** * @description 具体状态类A * @author zeling * @date 2018年1月14日 下午8:45:00 */
public class ConcreteStateA implements State { 
   

    @Override
    public void handle() {
        // TODO Auto-generated method stub
    }
}
package xyz.zeling.test.state.template;

import xyz.zeling.test.state.template.base.State;

/** * @description 具体状态类A * @author zeling * @date 2018年1月14日 下午8:45:00 */
public class ConcreteStateB implements State { 
   

    @Override
    public void handle() {
        // TODO Auto-generated method stub
    }
}
package xyz.zeling.test.state.template;

import xyz.zeling.test.state.template.base.State;

/** * @description 具体状态类A * @author zeling * @date 2018年1月14日 下午8:45:00 */
public class ConcreteStateC implements State { 
   

    @Override
    public void handle() {
        // TODO Auto-generated method stub
    }
}
package xyz.zeling.test.state.template;

import xyz.zeling.test.state.template.base.State;

/** * @description 状态模式,环境角色类 * @author zeling * @date 2018年1月14日 下午8:43:58 */
public class Context { 
   

    /** * 状态对象 */
    private State state;

    /** * @description 设置状态 * @date 2018年1月14日 下午9:13:20 * @param state 具体状态 */
    public void setState(State state) {
        this.state = state;
    }

    /** * @description 执行策略方法 * @date 2018年1月14日 下午8:43:31 */
    public void request() {
        state.handle();
    }
}

实例应用:橘子有不同颜色,状态不同的橘子颜色不同,用状态模式实现。

package xyz.zeling.test.state.demo.base;

/** * @description 状态模式,状态接口 * @author zeling * @date 2018年1月14日 下午4:13:21 */
public interface Orange { 
   
    /** * @description 输出橘子的颜色 * @date 2018年1月14日 下午4:14:00 */
    void printColor();
}
package xyz.zeling.test.state.demo;

import xyz.zeling.test.state.demo.base.Orange;

/** * @description 状态2:青色的橘子 * @author zeling * @date 2018年1月14日 下午4:17:52 */
public class CyanOrange implements Orange { 
   

    @Override
    public void printColor() {
        System.out.println("My color is cyan!");
    }
}
package xyz.zeling.test.state.demo;

import xyz.zeling.test.state.demo.base.Orange;

/** * @description 橘子状态管理器 * @author zeling * @date 2018年1月14日 下午4:19:55 */
public class OrangeStateManage { 
   

    /** * 橘子的状态 */
    private Orange state;

    /** * @description 设置橘子的状态 * @date 2018年1月14日 下午4:21:36 * @param state */
    public void setState(Orange state) {
        this.state = state;
    }

    /** * @description 输出当前状态的橘子颜色 * @date 2018年1月14日 下午4:22:08 */
    public void print() {
        state.printColor();
    }
}
package xyz.zeling.test.state.demo;

import xyz.zeling.test.state.demo.base.Orange;

/** * @description 状态1:红色外观的橘子 * @author zeling * @date 2018年1月14日 下午4:15:29 */
public class RedOrange implements Orange { 
   

    @Override
    public void printColor() {
        System.out.println("My color is red!");
    }
}
package xyz.zeling.test.state.demo;

import xyz.zeling.test.state.demo.base.Orange;

/** * @description 状态3:黄色的橘子 * @author zeling * @date 2018年1月14日 下午4:18:45 */
public class YellowOrange implements Orange { 
   

    @Override
    public void printColor() {
        System.out.println("My color is yellow!");
    }
}

客户端

    /** * @description 使用状态模式:状态模式,又称状态对象模式(Pattern of Objects for * States),状态模式是对象的行为模式。状态模式允许一个对象在其内部状态改变的时候改变其行为。这个对象看上去就像是改变了它的类一样 * @date 2018年1月14日 下午4:04:16 */
    public static void useState() {
        // 具体使用状态
        Orange cyanOrange = new CyanOrange();
        // 创建环境
        OrangeStateManage orangeStateManage = new OrangeStateManage();
        // 设置状态并执行
        orangeStateManage.setState(cyanOrange);
        orangeStateManage.print();
    }

策略模式和状态模式的比较

讲真,我觉得它们都差不多啊,好难区别啊!!!
不过,虽然讲不出来它们的区别是什么,但是有个例子可以很好的描述它们的区别

  • 状态模式:这个模式就好比员工申请离职单的流程,离职单到直接上级,这个状态就是直接上级批示,等直接上级审阅之后,通过了就到下一个状态。这一个个状态对应不同的处理,这是有顺序要求的。
  • 策略模式:这个模式好比于你假期要出国游玩,有日本、美国、新加坡等国家,你每到一个国家就执行不同的游玩策略,可以先去日本,也可以先去美国,没有顺序要求。

demo

github地址:https://github.com/zeling1005/state-and-strategy.git

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

发布者:全栈程序员-用户IM,转载请注明出处:https://javaforall.cn/126050.html原文链接:https://javaforall.cn

【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛

【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...

(0)


相关推荐

  • 计算机硬件知识

    一为何要学习计算机基础python是编程语言,即python是语言语言有英语、法语、葡萄牙语等,但凡是语言,都是用来沟通的介质。程序员编程的本质就是让计算机去工作,而编程语言就是程序员与计算机

  • win10 命令行进入指定目录方法[通俗易懂]

    win10 命令行进入指定目录方法[通俗易懂]原文地址:https://blog.csdn.net/weixin_29207913/article/details/106616959方法一在资源管理器”地址“前面输入cmd。如下图:确认后cmd就会进入相关目录,进行其他cmd操作。如下图:方法二在资源管理器中,按住shift,同时右键空白地方,将会出现分别点击红框中选项,能分别使用PowerShell和WSL进入对应目录…

    2022年10月16日
  • js用户管理中心tab切换界面模板

    效果体验:http://hovertree.com/texiao/js/27/效果图如下:代码如下:转自:http://hovertree.com/h/bjaf/hovertreepanel.htm

    2021年12月24日
  • mariadb 的安装及基本配置

    mariadb 的安装及基本配置文章目录一、mariadb介绍二、mariadb下载及安装三、mariadb的启停命令四、mariadb的配置五、添加用户,设置权限Navicat连接数据库一、mariadb介绍MariaDB数据库管理系统是MySQL的一个分支,主要由开源社区在维护,采用GPL授权许可。开发这个分支的原因之一是:甲骨文公司收购了MySQL后,有将MySQL闭源的潜在风险,因此社区采用分支的方式来避开这个风险。MariaDB的目的是完全兼容MySQL,包括API和命令行,使之能轻松成为MySQL的代替品。在存储引擎

  • js添加事件和移除事件:addEventListener()与removeEventListener()

    js添加事件和移除事件:addEventListener()与removeEventListener()作用:   addEventListener()与removeEventListener()用于处理指定和删除事件处理程序操作。   它们都接受3个参数:事件名、事件处理的函数和布尔值。   布尔值参数是true,表示在捕获阶段调用事件处理程序;如果是false,表示在冒泡阶段调用事件处理程序。示例:环境:移动端,界面禁止触摸事件要在bo

  • 基尼系数的计算原理是什么_基尼系数1

    基尼系数的计算原理是什么_基尼系数1理论基尼指数( GiniIndex )是20世纪初经济学家基尼定义的指标,最为知名的应用是考察居民收入的差异情况。居民收入的情况符合幂指函数( PowerLaw )分布,最直观(但非准确)的理解就是 80/20 原则,也就是 20%的人拥有了 80% 的人的财富。用公式表示就是描述了是收入靠后 %x 的人所拥有的收入总和占所有人收入总和的比例 f(x) 的关系。

    2022年10月13日

发表回复

您的电子邮箱地址不会被公开。

关注全栈程序员社区公众号