【设计模式系列】--策略模式

简介: 什么是策略模式 在前面的博文中,小编主要向小伙伴介绍了组合模式,今天这篇博文,我们继续来学习设计模式的相关知识,今天和小伙伴们见面的是策略模式,策略模式英文名字叫Strategy,策略模式属于行为模式的一种,她对一系列的算法加以封装,为所有算法定义一个抽象的算法接口,并通过继承该抽象算法接口对所有的算法加以封装和实现,具体的算法选择交由客户端决定,策略模式主要用来平滑的处理算法的切换。

什么是策略模式

在前面的博文中,小编主要向小伙伴介绍了组合模式,今天这篇博文,我们继续来学习设计模式的相关知识,今天和小伙伴们见面的是策略模式,策略模式英文名字叫Strategy,策略模式属于行为模式的一种,她对一系列的算法加以封装,为所有算法定义一个抽象的算法接口,并通过继承该抽象算法接口对所有的算法加以封装和实现,具体的算法选择交由客户端决定,策略模式主要用来平滑的处理算法的切换。

策略模式结构图

我们来看一下策略的结构图,如下所示:


  对上述结构图进行简单的解释说明:

        a、将所有的算法都抽象成了Strategy,可以将算法分离出来并且进行更换。
        b、Context 中含有对Strategy的引用。
        c、通过contextInterface(),进行对算法的使用。

         从上面的结构图中,可以看出这些算法完成的都是相同的工作,只是实现不同,它可以以相同的方式调用所有的算法,减少各种算法类与使用算法类之间的耦合。换句话说,策略模式并不将算法固定在具体的某个类中,而是将算法独立出来,可根据需要替换算法。例如:Context中含有对 Straategy的引用。这里还用到了依赖倒转和里斯代换原则,即Context依赖于抽象,而没有依赖具体的子类,并且,子类可以替换父类。

策略模式简单应用demo

接着,我们通过一个简单的demo来了解一下策略模式是如何在实际中加以应用的,新建java项目Strategy,新建类MainClass,新建接口Strategy,编写接口里面的代码部分,如下所示:

public interface Strategy {
	//加密
	public void encrypt();
	
新建类MD5Strategy,实现接口Strategy,编写相关代码,如下所示:

public class MD5Strategy implements Strategy {

	@Override
	public void encrypt() {
		System.out.println("执行MD5加密");

	}

}
新建类MDSStrategy,实现接口Strategy,编写相关代码,如下所示:

public class MDSStrategy implements Strategy {

	@Override
	public void encrypt() {
		System.out.println("执行MDS加密");

	}

}
如果我们不使用策略模式,我们会在MainClass里面如何编写代码呢?如下所示:

public class MainClass {
	public static void main(String[] args){
		Strategy stra =  new MD5Strategy();
		stra.encrypt();
	}
}
运行如下所示:



如果要执行MDSS的方法,我们可以对MainClass中的代码部分进行修改,如下所示:

public class MainClass {
	public static void main(String[] args){
		Strategy stra =  new MDSStrategy();
		stra.encrypt();
	}
}
运行效果如下所示:

通过这种方式,可以实现动态的转变,我们只需要在客户端进行改变即可,但是这个不符合策略模式,通过结构图,我们可以看到有一个Context,可以理解成是一个工厂,创建Context类,并编写相关代码,如下所示:

public class Context {
	private Strategy strategy;
	public Context(Strategy strategy){
		this.strategy = strategy;
	}
	
	public void encrypt(){
		this.strategy.encrypt();
	}

}
编写客户端的代码,如下所示:

public class MainClass {
	public static void main(String[] args){
//		Strategy stra =  new MDSStrategy();
//		stra.encrypt();
		
		Context context = new Context(new MD5Strategy());
		context.encrypt();
	}
}
运行如下所示:



执行MDSS,修改客户端代码如下所示:

public class MainClass {
	public static void main(String[] args){
//		Strategy stra =  new MDSStrategy();
//		stra.encrypt();
		
		Context context = new Context(new MDSStrategy());
		context.encrypt();
	}
}
运行如下所示:


通过这两种方式的实现,经过对比我们可以发现,我们直接通过Context进行调用,我们可以把Strategy看成是一个抽象的接口,我们再来举一个简单的例子,帮助我们进一步加深对策略模式的理解,新建包Strategy,新建类MainClass,新建接口Strategy,编写相关代码如下所示:

package Strategy;

public interface Strategy {
	public double cost(double num);
}
新建类StrategyA实现Strategy,编写相关代码,如下所示:

package Strategy;

public class StrategyA implements Strategy {

	@Override
	public double cost(double num) {
		return num*0.8;
	}

}
新建类Context,编写相关代码,如下所示:

package Strategy;

public class Context {
	private Strategy strategy;
	
	public Context (Strategy strategy){
		this.strategy=strategy;
	}
	
	public double cost(double num){
		return this.strategy.cost(num);
	}

}
新建MainClass,编写相关代码,如下所示:

package Strategy;

public class MainClass {
	public static void main(String[] args) {
		double num = 200;
		Context context = new Context(new StrategyA());
		double newNum = context.cost(num);
		System.out.println("实际付款"+newNum+"元");
	}

}
运行如下所示:


这个时候,商家改变策略,不打八折了,这个时候改成满200返50,我们该如何实现呢?新建类StrategyB,实现Strategy,编写相关代码,如下所示:

package Strategy;

public class StrategyB implements Strategy {

	@Override
	public double cost(double num) {
		if(num >= 200){
			return num-50;
		}
		return num;
	}

}
编写MainClass中的代码部分,如下所示:

package Strategy;

public class MainClass {
	public static void main(String[] args) {
		double num = 200;
		Context context = new Context(new StrategyB());
		double newNum = context.cost(num);
		System.out.println("实际付款"+newNum+"元");
	}

}
运行如下所示:


这样,我们直接实现接口,写一个方法就可以了,接着,我们来看一下策略模式的优缺点:

优点:

a、策略模式提供了管理相关的算法族的办法,策略类的等级结构定义了一个算法或行为族,恰当使用继承可以公共的代码移到父类里面,从而避免重复的代码。

b、策略模式提供了可以替换继承关系的办法,继承可以处理多种算法和行为,如果不使用策略模式,那么使用算法或行为的环境类就可能会有一些子类,每一个子类提供一个不同的算法和行为,但是,这样一来算法或行为的使用者就和算法或行为本身混在一起,决定使用哪一种算法或采取哪一种行为的逻辑就和算法或行为的逻辑混合在一起,从而不可能再单独演化,继承使得动态改变算法或行为变得不可能。

c、使用策略模式可以避免使用多重条件转移语句,多重转移语句不易维护,他把采用哪一种算法或采取哪一种行为的逻辑与算法或行为的逻辑混合在一起,统统列在一个多重转移语句里面,比使用继承的办法还要原始和落后。

缺点:

a、客户端必须知道所有的策略类,并执行决定使用哪一个策略类,这就意味着客户端必须理解这些算法的区别,以便适时选择恰当的算法类,换言之,策略模式只适用于客户端知道所有的算法或行为的情况。
b、策略模式造成很多的策略类,有时候可以通过把依赖于环境的状态保存到客户端里面,而将策略类设计成可共享的,这样策略类实例可以被不同客户端使用,换言之,可以使用享元模式来减少对象的数量。

小编寄语:该博文小编主要简单的介绍了策略模式,分别从什么是策略模式、策略模式的结构图、策略模式简单应用demo、策略模式的优缺点四个方面对策略模式进行了简单的介绍,策略模式是一个比较容易理解和使用的设计模式,策略模式是对算法的封装,它把算法的责任和算法本身分割开,委派给不同的对象管理。策略模式通常把一个系列的算法封装到一系列的策略类里面,作为一个抽象策略类的子类。用一句话来说,就是“准备一组算法,并将每一个算法封装起来,使得它们可以互换”。在策略模式中,应当由客户端自己决定在什么情况下使用什么具体策略角色。策略模式仅仅封装算法,提供新算法插入到已有系统中,以及老算法从系统中“退休”的方便,策略模式并不决定在何时使用何种算法,算法的选择由客户端来决定。这在一定程度上提高了系统的灵活性,但是客户端需要理解所有具体策略类之间的区别,以便选择合适的算法,这也是策略模式的缺点之一,在一定程度上增加了客户端的使用难度。


目录
相关文章
|
12天前
|
设计模式 人工智能 算法
基于多设计模式的状态扭转设计:策略模式与责任链模式的实战应用
接下来,我会结合实战案例,聊聊如何用「策略模式 + 责任链模式」构建灵活可扩展的状态引擎,让抽奖系统的状态管理从「混乱战场」变成「有序流水线」。
|
5月前
|
设计模式 算法 Java
设计模式觉醒系列(04)策略模式|简单工厂模式的升级版
本文介绍了简单工厂模式与策略模式的概念及其融合实践。简单工厂模式用于对象创建,通过隐藏实现细节简化代码;策略模式关注行为封装与切换,支持动态替换算法,增强灵活性。两者结合形成“策略工厂”,既简化对象创建又保持低耦合。文章通过支付案例演示了模式的应用,并强调实际开发中应根据需求选择合适的设计模式,避免生搬硬套。最后推荐了JVM调优、并发编程等技术专题,助力开发者提升技能。
|
11月前
|
设计模式 算法 Kotlin
Kotlin - 改良设计模式 - 策略模式
Kotlin - 改良设计模式 - 策略模式
131 4
|
5月前
|
设计模式 算法 搜索推荐
【设计模式】【行为型模式】策略模式(Strategy)
一、入门 什么是策略模式? 策略模式是一种行为设计模式,允许在运行时选择算法或行为。它将算法封装在独立的类中,使得它们可以互换,而不影响客户端代码。 为什么需要策略模式? 策略模式的主要目的是解决算法
106 14
|
8月前
|
设计模式 算法 开发者
「全网最细 + 实战源码案例」设计模式——策略模式
策略模式(Strategy Pattern)是一种行为型设计模式,用于定义一系列可替换的算法或行为,并将它们封装成独立的类。通过上下文持有策略对象,在运行时动态切换算法,提高代码的可维护性和扩展性。适用于需要动态切换算法、避免条件语句、经常扩展算法或保持算法独立性的场景。优点包括符合开闭原则、运行时切换算法、解耦上下文与策略实现、减少条件判断;缺点是增加类数量和策略切换成本。示例中通过定义抽象策略接口和具体策略类,结合上下文类实现动态算法选择。
239 8
「全网最细 + 实战源码案例」设计模式——策略模式
|
10月前
|
设计模式 存储 缓存
前端必须掌握的设计模式——策略模式
策略模式(Strategy Pattern)是一种行为型设计模式,旨在将多分支复杂逻辑解耦。每个分支类只关心自身实现,无需处理策略切换。它避免了大量if-else或switch-case代码,符合开闭原则。常见应用场景包括表单验证、风格切换和缓存调度等。通过定义接口和上下文类,策略模式实现了灵活的逻辑分离与扩展。例如,在国际化需求中,可根据语言切换不同的词条包,使代码更加简洁优雅。总结来说,策略模式简化了多条件判断,提升了代码的可维护性和扩展性。
|
11月前
|
设计模式 算法 Kotlin
Kotlin教程笔记(53) - 改良设计模式 - 策略模式
Kotlin教程笔记(53) - 改良设计模式 - 策略模式
99 1
|
11月前
|
设计模式 前端开发 JavaScript
JavaScript设计模式及其在实战中的应用,涵盖单例、工厂、观察者、装饰器和策略模式
本文深入探讨了JavaScript设计模式及其在实战中的应用,涵盖单例、工厂、观察者、装饰器和策略模式,结合电商网站案例,展示了设计模式如何提升代码的可维护性、扩展性和可读性,强调了其在前端开发中的重要性。
139 2
|
11月前
|
设计模式 算法 Kotlin
Kotlin教程笔记(53) - 改良设计模式 - 策略模式
Kotlin教程笔记(53) - 改良设计模式 - 策略模式
99 2
|
12月前
|
设计模式 算法 Kotlin
Kotlin教程笔记(53) - 改良设计模式 - 策略模式
本教程详细讲解Kotlin语法,适合深入学习。快速入门可参考“简洁”系列教程。本文通过游泳运动员的案例,介绍策略模式及其在Kotlin中的改良应用,利用高阶函数简化代码结构,提高灵活性。
112 3