策略模式

策略模式的定义是:定义了一系列的算法,把它们一个个的封装起来,并且使它们可相互替换,让算法可以独立于使用它的客户而变化。

设计原则是:把一个类中经常改变或者将来可能会经常改变的部分提取出来作为一个接口,然后在使用类中包含这个接口的实例,这样使用类的对象就可以随意调用实现了这个接口的类行为。

在策略模式中有如下几个角色:

环境角色(Context): 此角色中实现了对策略角色的引用和使用。

抽象策略角色:此角色通常由抽象类或接口来实现,定义了,所以具体策略角色的行为。

具体策略角色:此角色封装了实现不同功能的不同算法。

演示代码如下:

抽象策略类

public interface Strategy {
/**
* 策略方法
*/
void strategyMethod();
}

具体角色类A

public class ConcreteStrategyA implements Strategy{

    @Override
public void strategyMethod() {
//具体的行为
}
}

具体角色类B

public class ConcreteStrategyB implements Strategy {
@Override
public void strategyMethod() {
//具体的行为
}
}

环境角色类

public class Context {
/**
* 持有一个具体的策略对象
*/
private Strategy strategy; /**
* 构造方法,传入一个具体的策略对象
* @param strategy 具体的策略对象
*/
public Context(Strategy strategy)
{
this.strategy = strategy;
} /**
* 对外提供的使用策略的方法
*/
public void contextMethod()
{
//通常会转调具体的策略对象进行算法运算
strategy.strategyMethod();
}
}

策略模式具体场景例子

某Saas公司的企业服务系统,在销售时,会根据客户的购买时长来确定优惠策略,分为普通客户(无优惠政策),大客户(98折),战略客户(95折)。普通客户是指一次性租用服务在一到3年之间的,大客户指一次性使用服务3到5年之间的,战略客户指一次性使用服务5年以上的。因为每种客户价格算法不同,所以这个场景就可以使用策略模式。

定义一个计算价格行为的接口

public interface SalePrice {
/**
* 根据原价返回不同的价格
* @param originalPrice 原始价格
* @return
*/
BigDecimal salePrice(BigDecimal originalPrice);
}

然后定义三中客户的具体计算价格类(策略类)

public class OriginalCustomer implements SalePrice {
/**
* 普通客户直接返回原价
* @param originalPrice 原始价格
* @return 计算后的价格
*/
@Override
public BigDecimal salePrice(BigDecimal originalPrice) { return originalPrice.multiply(BigDecimal.valueOf(1));
}
}
public class LargeCustomer implements SalePrice {
/**
* 大客户返回98折价格
* @param originalPrice 原始价格
* @return 计算后的价格
*/
@Override
public BigDecimal salePrice(BigDecimal originalPrice) { return originalPrice.multiply(BigDecimal.valueOf(0.98));
}
}
public class StrategicCustomer implements SalePrice {
/**
* 战略客户直接返回95折价格
* @param originalPrice 原始价格
* @return 计算后的价格
*/
@Override
public BigDecimal salePrice(BigDecimal originalPrice) {
return originalPrice.multiply(BigDecimal.valueOf(0.95));
}
}

客户类,需要判断具体的调用哪个计算价格的方法

public class Customer {

    private int years;
/** 租用服务一年的初始价格 */
private BigDecimal originalPrice = BigDecimal.valueOf(50000);
/** 客户最终支付的价格 **/
private BigDecimal payForPrice = BigDecimal.ZERO;
/** 每个客户的初始价格都是原价 */
private SalePrice salePrice = new OriginalCustomer(); /**
* 根据客户购买的时长来计算每一年的优惠价格(单位:年)
* @param years
*/
public void buy(int years)
{
this.years = years;
payForPrice = originalPrice.multiply(BigDecimal.valueOf(years));
//大于5年的战略客户价格
if(years >= 5){
salePrice = new StrategicCustomer();
}else if(years >= 3)//3年到5年的大客户优惠价格
{
salePrice = new LargeCustomer();
}else if(years >= 1)//1到3年的普通用户原价
{
salePrice = new OriginalCustomer();
}
} /**
* 计算客户最终支付的价格
* @return
*/
public BigDecimal payPrice(){
return salePrice.salePrice(payForPrice);
} }

客户端调用类,自动计算支付价格

**
* 客户端调用类
*/
@Slf4j
public class Client { public static void main(String[] args){
Customer customer = new Customer(); customer.buy(1);
log.info("客户需支付:{}",customer.payPrice()); customer.buy(3);
log.info("客户需支付:{}",customer.payPrice()); customer.buy(6);
log.info("客户需支付:{}",customer.payPrice()); }
}

输出结果:

客户需支付:50000
客户需支付:147000.00
客户需支付:285000.00

根据输出结果可以看出购买不同时间的服务,收费价格是不同的。这样客户可以不用依赖具体的收费方法,直接根据需要的服务的时间购买即可。

Java设计模式学习记录-策略模式的更多相关文章

  1. Java设计模式学习记录-状态模式

    前言 状态模式是一种行为模式,用于解决系统中复杂的对象状态转换以及各个状态下的封装等问题.状态模式是将一个对象的状态从该对象中分离出来,封装到专门的状态类中,使得对象的状态可以灵活多变.这样在客户端使 ...

  2. Java设计模式学习记录-模板方法模式

    前言 模板方法模式,定义一个操作中算法的骨架,而将一些步骤延迟到子类中.使得子类可以不改变一个算法的结构即可重新定义该算法的某些特定步骤. 模板方法模式 概念介绍 模板方法模式,其实是很好理解的,具体 ...

  3. Java设计模式学习记录-备忘录模式

    前言 这次要介绍的是备忘录模式,也是行为模式的一种 .现在人们的智能手机上都会有备忘录这样一个功能,大家也都会用,就是为了记住某件事情,防止以后自己忘记了.那么备忘录模式又是什么样子的呢?是不是和手机 ...

  4. Java设计模式学习记录-迭代器模式

    前言 这次要介绍的是迭代器模式,也是一种行为模式.我现在觉得写博客有点应付了,前阵子一天一篇,感觉这样其实有点没理解透彻就写下来了,而且写完后自己也没有多看几遍,上次在面试的时候被问到java中的I/ ...

  5. Java设计模式学习记录-解释器模式

    前言 这次介绍另一个行为模式,解释器模式,都说解释器模式用的少,其实只是我们在日常的开发中用的少,但是一些开源框架中还是能见到它的影子,例如:spring的spEL表达式在解析时就用到了解释器模式,以 ...

  6. Java设计模式学习记录-命令模式

    前言 这次要介绍的是命令模式,这也是一种行为型模式.最近反正没有面试机会我就写博客呗,该投的简历都投了.然后就继续看书,其实看书也会给自己带来成就感,原来以前不明白的东西,书上已经给彻底的介绍清楚了, ...

  7. Java设计模式学习记录-外观模式

    前言 这次要介绍的是外观模式(也称为门面模式),外观模式也属于结构型模式,其实外观模式还是非常好理解的,简单的来讲就是将多个复杂的业务封装成一个方法,在调用此方法时可以不必关系具体执行了哪些业务,而只 ...

  8. Java设计模式学习记录-桥接模式

    前言 这次介绍结构型设计模式中的第二种模式,桥接模式. 使用桥接模式的目的就是为了解耦,松散的耦合更利于扩展,但是会增加相应的代码量和设计难度. 桥接模式 桥接模式是为了将抽象化与实现化解耦,让二者可 ...

  9. Java设计模式学习记录-代理模式

    代理模式 代理模式是常见设计模式的一种,代理模式的定义是:为其他对象提供一种代理以控制对这个对象的访问. 在某些情况下,一个对象不适合或者不能直接引用另一个对象,而代理对象可以在客户端和目标对象之间起 ...

随机推荐

  1. hdu 4927 组合+公式

    http://acm.hdu.edu.cn/showproblem.php?pid=4927 给定一个长度为n的序列a,每次生成一个新的序列,长度为n-1,新序列b中bi=ai+1−ai,直到序列长度 ...

  2. python处理excel之读:xlrd模块

    # -*- coding:utf-8 -*- import xlrd path = r'D:/工作簿1(已自动还原).xlsx' # 打开excel文件读取数据 data = xlrd.open_wo ...

  3. GetFileOpenName()、GetFilesavename

    GetFileOpenName() 功能显示打开文件对话框,让用户选择要打开的文件. 语法:GetFileOpenName(title,pathname,filename{,extension{,fi ...

  4. headpq

    从一个集合中获得最大或者最小的N个元素列表 http://python3-cookbook.readthedocs.io/zh_CN/latest/c01/p04_find_largest_or_sm ...

  5. mac下查看jdk安装版本及安装目录

    使用IntelliJ idea新建工程时需要查看jdk安装目录,记录下来为以后备用. mac自带jdk,查看jdk版本: IcarusdeMacBook-Pro:~ icarus$ java -ver ...

  6. CF1146H Satanic Panic

    题目传送门 Description 给定二维平面内\(n\)个点\((n\leq 300)\),求能组成五角星(不要求正五角星)的五元组个数. Solution 一道小清新的寄蒜几盒计算几何题,代码不 ...

  7. [ActionScript 3.0] as3处理xml的功能和遍历节点

    as3比as2处理xml的功能增强了N倍,获取或遍历节点非常之方便,类似于json对像的处理方式. XML 的一个强大功能是它能够通过文本字符的线性字符串提供复杂的嵌套数据.将数据加载到 XML 对象 ...

  8. 在Ubuntu下编译安装nginx

    一.安装nginx 1.安装前提 a)epoll,linux内核版本为2.6或者以上 b)gcc编译器,g++编译器 c)pcre库,函数库,支持解析正则表达式 d)zlib库:压缩解压功能 e)op ...

  9. Docker三剑客之Docker Compose

    一.什么是Docker Compose Compose 项目是Docker官方的开源项目,负责实现Docker容器集群的快速编排,开源代码在https://github.com/docker/comp ...

  10. Python全局解释器锁 -- GIL

    首先强调背景: 1.GIL是什么?GIL的全称是Global Interpreter Lock(全局解释器锁),来源是python设计之初的考虑,为了数据安全所做的决定. 2.每个CPU在同一时间只能 ...