专栏首页波波烤鸭Java策略模式(Strategy)

Java策略模式(Strategy)

策略模式(Strategy)

一.不使用策略模式

使用场景: 某个市场人员接到单后的报价策略(CRM系统中常见问题)。报价策略很复杂,可以简单作如下分类:

  1. 普通客户小批量报价
  2. 普通客户大批量报价
  3. 老客户小批量报价
  4. 老客户大批量报价

具体选用哪个报价策略,这需要根据实际情况来确定。这时候,如果我们用普通的方式来实现如下

public double getPrice(String type, double price) {
	if (type.equals("普通客户小批量")) {
		System.out.println("不打折,原价");
		return price;
	} else if (type.equals("普通客户大批量")) {
		System.out.println("打九折");
		return price * 0.9;
	} else if (type.equals("老客户小批量")) {
		System.out.println("打八五折");
		return price * 0.85;
	} else if (type.equals("老客户大批量")) {
		System.out.println("打八折");
		return price * 0.8;
	}
	return price;
}

  这种通过条件判断的方式来实现比较符合普通程序员的思维习惯,但是算法比较复杂时,整个条件控制代码会变得很长,难于维护。

二.使用策略模式

  上面的案例就非常适合使用策略模式。策略模式对应于解决某一个问题的一个算法族,允许用户从该算法族中任选一个算法解决某一问题,同时可以方便的更换算法或者增加新的算法。并且由客户端决定调用哪个算法。我们通过案例来说明

1.案例类图

2.案例实现

Strategy接口

public interface Strategy {
	public double getPrice(double  standardPrice);
}

四种算法实现

public class NewCustomerFewStrategy implements Strategy {
	@Override
	public double getPrice(double standardPrice) {
		System.out.println("不打折,原价");
		return standardPrice;
	}
}
public class NewCustomerManyStrategy implements Strategy {
	@Override
	public double getPrice(double standardPrice) {
		System.out.println("打九折");
		return standardPrice*0.9;
	}
}
public class OldCustomerFewStrategy implements Strategy {
	@Override
	public double getPrice(double standardPrice) {
		System.out.println("打八五折");
		return standardPrice*0.85;
	}
}
public class OldCustomerManyStrategy implements Strategy {
	@Override
	public double getPrice(double standardPrice) {
		System.out.println("打八折");
		return standardPrice*0.8;
	}
}

对外统一出口

/**
 * 负责和具体的策略类交互
 * 这样的话,具体的算法和直接的客户端调用分离了,使得算法可以独立于客户端独立的变化。
 * 如果使用spring的依赖注入功能,还可以通过配置文件,动态的注入不同策略对象,动态的切换不同的算法.
 * @author Administrator
 *
 */
public class Context {
	//当前采用的算法对象
	private Strategy strategy;	

	//可以通过构造器来注入
	public Context(Strategy strategy) {
		super();
		this.strategy = strategy;
	}
	//可以通过set方法来注入
	public void setStrategy(Strategy strategy) {
		this.strategy = strategy;
	}
	public void pringPrice(double s){
		System.out.println("您该报价:"+strategy.getPrice(s));
	}
}

测试

public static void main(String[] args) {
	Strategy s1 = new OldCustomerManyStrategy();
	Context ctx = new Context(s1);
	ctx.pringPrice(998);
}

输出

打八折
您该报价:798.4000000000001

3.总结

  通过案例我们能清楚的看到,如果我们想要添加新的算法只需要再单独添加Strategy的实现就可以了,和其他算法没有影响。移除是也同样的道理,而且和客户端实现了分离。 本质:分离算法,选择实现。

4.开发中的使用场景

  1. JAVASE中GUI编程中,布局管理
  2. Spring框架中,Resource接口,资源访问策略
  3. javax.servlet.http.HttpServlet#service()

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

我来说两句

0 条评论
登录 后参与评论

相关文章

  • Java原型模式(prototype)

      prototype模式也就是原型模式,是javaGOF23种设计模式中的一种,我们在学习spring的时候在bean标签的学习中碰到过,所以本文来给大家介绍...

    用户4919348
  • IO模型介绍

      前面介绍了NIO中的buffer和Channel,而我们将NIO主要的使用场景还是在网络环境中,在具体介绍之前我们需要了解下IO的模型

    用户4919348
  • Spring之Aware接口介绍

      在Bean对象的生命周期的方法中有好几个接口是Aware接口的子接口,所以弄清楚Aware接口对于理解Spring框架还是很有帮助的。

    用户4919348
  • 设计模式之五(策略模式)

    策略模式:它定义了算法家族,分别封装起来,让它们之间可以相互替换,此模式让算法的变化,不会影响到使用算法的客户。

    aehyok
  • Spring bean的生命周期

    Spring中Bean的管理是其最基本的功能,根据下面的图来了解Spring中Bean的生命周期:

    用户3467126
  • 一种常见的关于率指标的错误分析思路

    今天给大家分享一种在数据分析过程中关于率指标分析可能会犯的一种错误。这个问题其实很多新人都会犯,有的老人也会犯,而且很多时候错了以后并不自知。刚好读者群有人在问...

    张俊红
  • 这些年,我们一直追逐的漏洞利用神器

    现在,也许我们所认为的漏洞利用工具神器,十年后,又该会是什么样子呢?可能大概就像我们现在看到以前的啊d注入工具的样子吧。

    Bypass
  • Appium:轻松玩转app+webview混合应用自动化测试

    Appium这个听起来既生疏也熟悉的自动化测试工具,比起原生的UiAutomator可能是异常的不起眼,可是却是有自身独当一面的能力,可以完成许多高难度作业,完...

    腾讯移动品质中心TMQ
  • Oracle 12c远程克隆PDB的问题及修复(r12笔记第78天)

    Oracle 12c里面的PDB迁移还是有很多花样的,玩法很多,如果想达到一种平滑方式的迁移,克隆远程PDB也是一种方法,保证网络畅通,即可远程克隆PDB到指...

    jeanron100
  • 听说有了这些黑科技,今年的超级碗才请到了Lady Gaga

    镁客网

扫码关注云+社区

领取腾讯云代金券