[设计模式] 策略模式


目录



手机用户请​​横屏​​​获取最佳阅读体验,​​REFERENCES​​中是本文参考的链接,如需要链接和更多资源,可以关注其他博客发布地址。


平台

地址



简书

​https://www.jianshu.com/u/3032cc862300​

个人博客

​https://yiyuery.github.io/NoteBooks/​


策略模式定义了算法族,分别封装起来,让他们之间可以互相替换,此模式让算法的变化独立于使用算法的客户。


涉及到的设计原则:


  • 多用组合,少用继承
  • 针对接口编程,而不是对实现编程
  • 找出应用中可能需要变化之处,把它们独立出来,不要和那些不需要变化的代码混合在一起

场景分析

假设我们现在需要定义一个手机的基本能力,call和message,分别表示通话和短信能力。来给客户演示不同手机的能力区别,以达到推销赞助商手机的功能。

但是随着智能手机的发展,手机的通话和短信的能力都已得到很大程度的增强。

比方说,我们有两款手机,一个是10年前的老牌 Nokia,智能发些简单的文本信息,打电话什么的也没有视频的能力。但是新版的小米手机,却能安装很多APP,比方说QQ、微信,不仅可以进行视频通话,还可以发送富文本信息。

那么问题来了,如何定义一个模型,能够将手机通话和短信的能力抽象出来,以满足今天我们需要用Nokia手机演示功能,明天我们要用小米手机演示功能,后天我们要…

这种时候,我们可以考虑很多种方案来设计,但是一个优先的模型设计应该是要尽可能的符合设计原则的。这不仅仅是编程技术发展过程中的前辈总结出来的一套武功秘籍,还是我们可以奉之为编程行为准则的优秀指导手册。

错误的或是更为复杂的功能设计不再赘述,仅在此处分享如何利用策略模式来实现我们本次的场景需求分析。

实战


首先对于手机的基本能力,我们定义抽象接口,一个是通话的策略接口,一个是短信的策略接口。


public interface ICallStrategy {

/**
* 简单的通话
*/
void simplyCall();

/**
* 视频通话
*/
void videoCall();
}

public interface IMessageStrateggy {

/**
* 简单的文本短信
*/
void simpleMessage();

/**
* 图文并茂的富文本短信
*/
void richTextMessage();
}

在接口中我们定义了通话的策略抽象,提供了两种行为:视频通话和普通通话;还定义了短信的策略抽象,也提供两种行为:简单文本短信和富文本短信。

由于我们要设计的模型是针对手机作为讨论主题的,无论是Nokia还是Xiaomi,我们很容易想到都是手机,可以采用继承的思想来实现功能。

所以我们定义一个抽象基类,用于定义手机的公共属性。

public abstract class BasePhone {

/**
* 通话
*/
protected ICallStrategy callStrategy;

/**
* 短信
*/
protected IMessageStrateggy messageStrateggy;

/**
* 打电话
*/
public abstract void call();

/**
* 发简讯
*/
public abstract void message();


/**
* 动态调整通话策略
* @param callStrategy
*/
public void setCallStrategy(ICallStrategy callStrategy) {
this.callStrategy = callStrategy;
}

/**
* 动态调整短信策略
* @param messageStrateggy
*/
public void setMessageStrateggy(IMessageStrateggy messageStrateggy) {
this.messageStrateggy = messageStrateggy;
}
}

在基类中,为了避免不同手机使用不同的APP实现视频通话,或是富文本短信的发送的能力。我们采用策略类和手机基类组合的形式来完成模块设计。

如果在这里仅仅使用继承,直接在继承的子类中实现这些能力,当然也可以实现,但是会有问题,假如我们有几十个品牌的手机,在演示时各个手机都可以使用不同的APP来通话和发短信,但是,如果一款APP在某个手机中不支持,但是演示时却因为赞助商的原因一定要演示怎么办?

我们是不是就得定义两个相同APP的行为类,一个是支持通话和短信,一个是不支持通话和短信。然后在该手机实现类中创建不支持的实例,调用对应的call和message方法来提示用户不支持?

这也是继承的一个弊端,代码模块之间耦合比较严重,子类往往需要继承父类的所有属性,无论有用没有。

使用组合的话,我们可以把使用哪个APP进行通话和短信的行为抽象成策略,在实现的子类中定义一族该对象支持的策略,在根据策略族中是否含有我们要求的行为来判断设备是否支持,而非在多个子类中通过修改代码或是覆盖父类方法来实现功能。


Xiaomi手机


[设计模式] 策略模式_设计模式

public class XiaomiPhone extends BasePhone {

/**
* 打电话
*/
@Override
public void call() {
//可以是视频电话、也可以是默认的通话,视频通话还可以由不同的APP应用软件发起
callStrategy.videoCall();
}

/**
* 发简讯
*/
@Override
public void message() {
//可以使短信,也可以是QQ、WeChat、或是其他聊天工具发起
messageStrateggy.richTextMessage();
}
}


老版 Nokia
[设计模式] 策略模式_iphone_02


public class SimpleNokiaPhone extends BasePhone {

/**
* 打电话
*/
@Override
public void call() {
System.out.println("I can make a call!");
}

/**
* 发简讯
*/
@Override
public void message() {
System.out.println("I can send a simple message!");
}

@Override
public void setCallStrategy(ICallStrategy callStrategy) {
throw new IllegalArgumentException("not support");
}

@Override
public void setMessageStrateggy(IMessageStrateggy messageStrateggy) {
throw new IllegalArgumentException("not support");
}
}

在Nokia中,我们知道他不能安装APP,所以只有基本的通话和短信方式,我们在设置行为能力的策略的时候,可以让其对外抛出异常,这样上层就知道该手机不能用对应APP进行短信和通话了。


接下来,我们定义个QQ和WeChat两款常用的软件来描述我们的手机能力。


QQ

public class QQMessageStrategy  implements IMessageStrateggy {
/**
* 简单的文本短信
*/
@Override
public void simpleMessage() {
System.out.println("QQ simple text message is sending...");
}

/**
* 图文并茂的富文本短信
*/
@Override
public void richTextMessage() {
//发送表情包
System.out.println("QQ Emoji Pack is sending...");
}
}

public class QQCallStrategy implements ICallStrategy {

/**
* 简单的通话
*/
@Override
public void simplyCall() {
System.out.println("QQ is trying to make voice calling!");
}

/**
* 视频通话
*/
@Override
public void videoCall() {
System.out.println("QQ is trying to start a new Video Call....");
}
}

[设计模式] 策略模式_iphone_03

[设计模式] 策略模式_视频通话_04

WeChat

public class WeChatCallStrategy implements ICallStrategy {
/**
* 简单的通话
*/
@Override
public void simplyCall() {
System.out.println("not support!");
}

/**
* 视频通话
*/
@Override
public void videoCall() {
System.out.println("WeChat is trying to start a new Video Call....");
}
}

微信的话不定义短信策略,验证空指针异常。


接下来我们开始演示喽!


  • 首先有请我们古老的Nokia出场:
/**
* 测试老版 Nokia 短信和通话能力
*/
@Test
public void testX2(){
SimpleNokiaPhone nokiaPhone = new SimpleNokiaPhone();
nokiaPhone.message();
nokiaPhone.call();
nokiaPhone.setMessageStrateggy(new QQMessageStrategy());
}
I can send a simple message!
I can make a call!

java.lang.IllegalArgumentException: not support

at com.example.template.pattern.strategy.SimpleNokiaPhone.setMessageStrateggy(SimpleNokiaPhone.java:53)

可以看到在设置QQ这个APP的策略能力的时候抛出了异常。

  • 然后是我们优秀的小米手机:

它可以用QQ发短信和打视频电话哦!

/**
* 测试小米手机发送表情包短信和视频电话
*/
@Test
public void testX1(){
XiaomiPhone xiaomiPhone = new XiaomiPhone();
xiaomiPhone.setMessageStrateggy(new QQMessageStrategy());
xiaomiPhone.message();
xiaomiPhone.call();
}
QQ Emoji Pack is sending...

java.lang.NullPointerException
at com.example.template.pattern.strategy.XiaomiPhone.call(XiaomiPhone.java:34)

额,忘记告诉它打电话用什么软件了。

xiaomiPhone.setCallStrategy(new QQCallStrategy());
QQ Emoji Pack is sending...
QQ is trying to start a new Video Call....

竟然可以发表情包,还能视频通话,不愧是国产手机中的经典战斗机。

微信的空指针异常演示已经不小心通过QQ的能力演示出现了,就不再赘述了。

讲到这里,如果我们的赞助商变成了阿里巴巴,那么我们是不是得用支付宝或是淘宝演示下短信能力呢?

那就简单了,在定义个支付宝对应的策略类就行了。至此,我们实现了一开始的场景需求,无论是何种方式的通话和短信方式的功能演示,我们都只需要对策略类和手机进行扩展,就可以了。

​敲黑板,dadada...​

回顾下我们使用的设计原则:


  • 我们采用了继承和组合集合的方式
  • 我们封装了变化的部分:手机的通话和短信方式,取决于APP的选择
  • 我们面向接口和超类编程,定义了一些抽象的接口和抽象基类。
  • 这种定义算法族的,分别封装起来,让他们之间可以互相依赖,算法的变化独立于使用算法的客户的模式,我们称之为​​策略模式​

REFERENCES


《Head First》读书笔记


更多


扫码关注​​架构探险之道​​,回复文章标题,获取本文相关源码和资源链接


[设计模式] 策略模式_视频通话_05