使用桥接模式设计复杂的消息系统
本文节选自《设计模式就该这样学》
举个例子,我们在平时办公的时候经常通过邮件消息、短信消息或者系统内消息与同事进行沟通。尤其在走一些审批流程的时候,我们需要记录这些过程以备查。根据类型来划分,消息可以分为邮件消息、短信消息和系统内消息。但是,根据紧急程度来划分,消息可以分为普通消息、加急消息和特急消息。显然,整个消息系统可以划分为两个维度,如下图所示。
如果我们用继承,则情况就复杂了,而且也不利于扩展。邮件消息可以是普通的,也可以是加急的;短信消息可以是普通的,也可以是加急的。下面我们用桥接模式来解决这个问题。 首先创建一个IMessage接口担任桥接的角色。
```java
/* * 实现消息发送的统一接口 / public interface IMessage { //要发送的消息的内容和接收人 void send(String message, String toUser); }
```
创建邮件消息实现EmailMessage类。
```java
/* * 邮件消息的实现类 / public class EmailMessage implements IMessage { public void send(String message, String toUser) { System.out.println("使用邮件消息发送" + message + "给" + toUser); } }
```
创建短信消息实现SmsMessage类。
```java
/* * 短信消息的实现类 * SMS(Short IMessage Service)短信消息服务 / public class SmsMessage implements IMessage { public void send(String message, String toUser) { System.out.println("使用短信消息发送" + message + "给" + toUser); } }
```
然后创建桥接抽象角色AbstractMessage类。
```java
/* * 抽象消息类 / public abstract class AbstractMessage { //持有一个实现部分的对象 IMessage message;
//构造方法,传入实现部分的对象
public AbstractMessage(IMessage message) {
this.message = message;
}
//发送消息,委派给实现部分的方法
public void sendMessage(String message, String toUser) {
this.message.send(message, toUser);
}
}
```
创建具体实现普通消息NomalMessage类。
```java
/* * 普通消息类 / public class NomalMessage extends AbstractMessage {
//构造方法,传入实现部分的对象
public NomalMessage(IMessage message) {
super(message);
}
@Override
public void sendMessage(String message, String toUser) {
//对于普通消息,直接调用父类方法发送消息即可
super.sendMessage(message, toUser);
}
}
```
创建具体实现加急消息UrgencyMessage类。
```java
/* * 加急消息类 / public class UrgencyMessage extends AbstractMessage {
//构造方法
public UrgencyMessage(IMessage message) {
super(message);
}
@Override
public void sendMessage(String message, String toUser) {
message = "加急:" + message;
super.sendMessage(message, toUser);
}
//扩展它功能,监控某个消息的处理状态
public Object watch(String messageId) {
//根据给出的消息编码(messageId)查询消息的处理状态
//组织成监控的处理状态,然后返回
return null;
}
}
```
最后编写客户端测试代码。
```java
public static void main(String[] args) { IMessage message = new SmsMessage(); AbstractMessage abstractMessage = new NomalMessage(message); abstractMessage.sendMessage("加班申请速批", "王总");
message = new EmailMessage();
abstractMessage = new UrgencyMessage(message);
abstractMessage.sendMessage("加班申请速批", "王总");
}
```
运行结果如下图所示。
在上面的案例中,我们采用桥接模式解耦了“消息类型”和“消息紧急程度”这两个独立变化的维度。后续如果有更多的消息类型,比如微信、钉钉等,则直接新建一个类继承IMessage即可;如果紧急程度需要新增,则同样只需新建一个类实现AbstractMessage类即可。
关注『 Tom弹架构 』回复“设计模式”可获取完整源码。
【推荐】Tom弹架构:30个设计模式真实案例(附源码),挑战年薪60W不是梦
本文为“Tom弹架构”原创,转载请注明出处。技术在于分享,我分享我快乐!如果本文对您有帮助,欢迎关注和点赞;如果您有任何建议也可留言评论或私信,您的支持是我坚持创作的动力。关注『 Tom弹架构 』可获取更多技术干货!
- 使用桥接模式设计复杂的消息系统
- 为什么MySQL索引结构采用B 树?
- 为什么Netty线程池默认大小为CPU核数的2倍
- 谈谈你对深克隆和浅克隆的理解
- 什么是代理,为什么要用动态代理?
- 什么是零拷贝,Netty是如何实现的?
- 3分钟轻松理解单线程下的HashMap工作原理
- 被面试官问烂的Spring AOP原理,你是怎么答的?
- Spring为何需要三级缓存解决循环依赖,而不是二级缓存?
- 为什么Spring中每个Bean都要定义作用域?
- 谈谈你对Spring Bean的理解
- 趣谈装饰器模式,让你一辈子不会忘
- 掌握这些招数,你也能写出HR眼中的高分简历
- MongoDB高级应用之数据转存与恢复(5)
- 图解MongoDB集群部署原理(3)
- 爆肝30天,肝出来史上最透彻Spring原理和27道高频面试题总结
- Spring核心原理之IoC容器初体验(2)
- Spring核心原理分析之MVC九大组件(1)
- 30个类手写Spring核心原理之动态数据源切换(8)
- 【紧急】Log4j又发新版2.17.0,只有彻底搞懂漏洞原因,才能以不变应万变,小白也能看懂