重学Java设计模式-结构型模式-装饰器模式
重学Java设计模式-结构型模式-装饰器模式
内容摘自:https://bugstack.cn/md/develop/design-pattern/2020-06-09-重学 Java 设计模式《实战装饰器模式》.html#重学-java-设计模式-实战装饰器模式「sso单点登录功能扩展-增加拦截用户访问方法范围场景」
装饰器模式介绍#
初看上图感觉装饰器模式有点像俄罗斯套娃、某众汽车🚕,而装饰器的核心就是再不改原有类的基础上给类新增功能。不改变原有类,可能有的小伙伴会想到继承、AOP切面,当然这些方式都可以实现,但是使用装饰器模式会是另外一种思路更为灵活,可以避免继承导致的子类过多,也可以避免AOP带来的复杂性。
你熟悉的场景很多用到装饰器模式
new BufferedReader(new FileReader(""));
,这段代码你是否熟悉,相信学习java开发到字节流、字符流、文件流的内容时都见到了这样的代码,一层嵌套一层,一层嵌套一层,字节流转字符流等等,而这样方式的使用就是装饰器模式的一种体现。
案例场景模拟#
在本案例中我们模拟一个单点登录功能扩充的场景
一般在业务开发的初期,往往内部的ERP使用只需要判断账户验证即可,验证通过后即可访问ERP的所有资源。但随着业务的不断发展,团队里开始出现专门的运营人员、营销人员、数据人员,每个人员对于ERP的使用需求不同,有些需要创建活动,有些只是查看数据。同时为了保证数据的安全性,不会让每个用户都有最高的权限。
那么以往使用的SSO
是一个组件化通用的服务,不能在里面添加需要的用户访问验证功能。这个时候我们就可以使用装饰器模式,扩充原有的单点登录服务。但同时也保证原有功能不受破坏,可以继续使用。
1. 场景模拟工程#
itstack-demo-design-9-00
└── src
└── main
└── java
└── org.itstack.demo.design
├── HandlerInterceptor.java
└── SsoInterceptor.java
- 这里模拟的是spring中的类:
HandlerInterceptor
,实现接口功能SsoInterceptor
模拟的单点登录拦截服务。 - 为了避免引入太多spring的内容影响对设计模式的阅读,这里使用了同名的类和方法,尽可能减少外部的依赖。
2. 场景简述#
2.1 模拟Spring的HandlerInterceptor#
public interface HandlerInterceptor {
boolean preHandle(String request, String response, Object handler);
}
- 实际的单点登录开发会基于;
org.springframework.web.servlet.HandlerInterceptor
实现。
2.2 模拟单点登录功能#
public class SsoInterceptor implements HandlerInterceptor{
public boolean preHandle(String request, String response, Object handler) {
// 模拟获取cookie
String ticket = request.substring(1, 8);
// 模拟校验
return ticket.equals("success");
}
}
- 这里的模拟实现非常简单只是截取字符串,实际使用需要从
HttpServletRequest request
对象中获取cookie
信息,解析ticket
值做校验。 - 在返回的里面也非常简单,只要获取到了
success
就认为是允许登录。
装饰器模式重构代码#
接下来使用装饰器模式来进行代码优化,也算是一次很小的重构。
装饰器主要解决的是直接继承下因功能的不断横向扩展导致子类膨胀的问题,而是用装饰器模式后就会比直接继承显得更加灵活同时这样也就不再需要考虑子类的维护。
在装饰器模式中有四个比较重要点抽象出来的点;
- 抽象构件角色(Component) -
定义抽象接口
- 具体构件角色(ConcreteComponent) -
实现抽象接口,可以是一组
- 装饰角色(Decorator) -
定义抽象类并继承接口中的方法,保证一致性
- 具体装饰角色(ConcreteDecorator) -
扩展装饰具体的实现逻辑
通过以上这四项来实现装饰器模式,主要核心内容会体现在抽象类的定义和实现上。
1. 工程结构#
itstack-demo-design-9-02
└── src
└── main
└── java
└── org.itstack.demo.design
├── LoginSsoDecorator.java
└── SsoDecorator.java
装饰器模式模型结构
- 以上是一个装饰器实现的类图结构,重点的类是
SsoDecorator
,这个类是一个抽象类主要完成了对接口HandlerInterceptor
继承。 - 当装饰角色继承接口后会提供构造函数,入参就是继承的接口实现类即可,这样就可以很方便的扩展出不同功能组件。
2. 代码实现#
2.1 抽象类装饰角色#
public abstract class SsoDecorator implements HandlerInterceptor {
private HandlerInterceptor handlerInterceptor;
private SsoDecorator(){}
public SsoDecorator(HandlerInterceptor handlerInterceptor) {
this.handlerInterceptor = handlerInterceptor;
}
public boolean preHandle(String request, String response, Object handler) {
return handlerInterceptor.preHandle(request, response, handler);
}
}
- 在装饰类中有两个重点的地方是;1)继承了处理接口、2)提供了构造函数、3)覆盖了方法
preHandle
。 - 以上三个点是装饰器模式的核心处理部分,这样可以踢掉对子类继承的方式实现逻辑功能扩展。
2.2 装饰角色逻辑实现#
public class LoginSsoDecorator extends SsoDecorator {
private Logger logger = LoggerFactory.getLogger(LoginSsoDecorator.class);
private static Map<String, String> authMap = new ConcurrentHashMap<String, String>();
static {
authMap.put("huahua", "queryUserInfo");
authMap.put("doudou", "queryUserInfo");
}
public LoginSsoDecorator(HandlerInterceptor handlerInterceptor) {
super(handlerInterceptor);
}
@Override
public boolean preHandle(String request, String response, Object handler) {
boolean success = super.preHandle(request, response, handler);
if (!success) return false;
String userId = request.substring(8);
String method = authMap.get(userId);
logger.info("模拟单点登录方法访问拦截校验:{} {}", userId, method);
// 模拟方法校验
return "queryUserInfo".equals(method);
}
}
- 在具体的装饰类实现中,继承了装饰类
SsoDecorator
,那么现在就可以扩展方法;preHandle
- 在
preHandle
的实现中可以看到,这里只关心扩展部分的功能,同时不会影响原有类的核心服务,也不会因为使用继承方式而导致的多余子类,增加了整体的灵活性。
3. 测试验证#
3.1 编写测试类#
@Test
public void test_LoginSsoDecorator() {
LoginSsoDecorator ssoDecorator = new LoginSsoDecorator(new SsoInterceptor());
String request = "1successhuahua";
boolean success = ssoDecorator.preHandle(request, "ewcdqwt40liuiu", "t");
System.out.println("登录校验:" + request + (success ? " 放行" : " 拦截"));
}
- 这里测试了对装饰器模式的使用,通过透传原有单点登录类
new SsoInterceptor()
,传递给装饰器,让装饰器可以执行扩充的功能。 - 同时对于传递者和装饰器都可以是多组的,在一些实际的业务开发中,往往也是由于太多类型的子类实现而导致不易于维护,从而使用装饰器模式替代。
3.2 测试结果#
23:50:50.796 [main] INFO o.i.demo.design.LoginSsoDecorator - 模拟单点登录方法访问拦截校验:huahua queryUserInfo
登录校验:1successhuahua 放行
Process finished with exit code 0
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 分享4款.NET开源、免费、实用的商城系统
· 全程不用写代码,我用AI程序员写了一个飞机大战
· MongoDB 8.0这个新功能碉堡了,比商业数据库还牛
· 白话解读 Dapr 1.15:你的「微服务管家」又秀新绝活了
· 记一次.NET内存居高不下排查解决与启示