Spring 中使用责任链设计模式

本片文章不是讲Spring源码中使用责任链的设计实例,而是会讲在Spring中如何设计自己的责任链并且如何调用。

责任链设计模式作为我们常用的设计模式之一,用途非常的广,例如在一些流程化的执行中、或者是一些动态拦截中我们都可以使用责任链设计模式进行设计需求,从而使我们的项目无论是可用性还是可扩展性都会非常的好。

大家对于责任链还有不了解的可以看我之前的博文设计模式——责任链模式。

如何定义链条

在Java开发中我们如果想要自己定义一个链条,其实非常简单,也就定义一个抽象类,然后定义几个实现方法,然后设置其next属性即可,但是在Spring中如何将框架与我们的链条结合起来呢?其实这里用到了三个注解@Component、@Order、@PostConstruct。

  • @Component:将我们的子类交给Spring管理
  • @Order:定义我们链条的顺序
  • @PostConstruct:程序启动时将我们链条组合起来

接下来我们直接看代码,首先来看抽象类,其实责任链的抽象类基本上都一样的。

public abstract class PrintChainPattern {

 private PrintChainPattern next;

 public final void print() {
 String message = getMessage();

 log.info("{} : {}",message,message);
 if (getNext()!=null){
 getNext().print();
 }
 }
 public abstract String getMessage();
}

然后我们看实现类,后面有四个实现类,依次返回是two、three、four。@Order注解中数字依次递增。这里只演示第一个实现类的代码。

@Order(1)
@Component
public class OnePrintChainPattern extends PrintChainPattern{
 @Override
 public String getMessage() {
 return "one";
 }
}

接下来就到了如何利用Spring来组装我们的链条了

@Configuration
public class InitPrintChainPattern {

 @Autowired
 private List<PrintChainPattern> printChainPatterns;

 @PostConstruct
 private void initPrintChainPattern(){
 Collections.sort(printChainPatterns, AnnotationAwareOrderComparator.INSTANCE);

 int size = printChainPatterns.size();
 for (int i = 0; i < size; i++) {
 if (i == size-1){
 printChainPatterns.get(i).setNext(null);
 }else {
 printChainPatterns.get(i).setNext(printChainPatterns.get(i+1));
 }
 }
 }

 public void print(int index){
 printChainPatterns.get(index-1).print();
 }
}

这里我们可以看到在@PostConstruct 方法中我们做了两件事

  1. 将List<PrintChainPattern>中按照@Order注解的数字进行排序
  2. 依次设置每个节点的next值

这样我们就已经将这个链条组合了起来。接下来我们就可以随意的对这个链条进行操作,例如我下面的print()方法中,就是根据传进来的值的不同会从不同的节点进行执行。

如何在抽象类中使用@Autowired

在上面我们已经将我们链条组合了起来,但是如果我们的所有子类都公有一些类的话,那么这个类就要放在抽象类中。那么如果这个类我们想要从Spring的容器中取得呢?

比如我们有如下的类交给了Spring管理,我们所有子类都要使用这个类。

@Bean
public User setUser(){
 return User.builder().name("张三").age(14).build();
}

只需要在抽象类中定义一次即可,只需要在set方法上加@Autowired注解就能够将Spring容器中的类给注入进来。

private User user;

@Autowired
public void setUser(User user){
 this.user = user;
}

然后在子类中直接调用getUser()方法就行

@Override
public String getMessage() {
 log.info("name:{},age:{}",getUser().getName(),getUser().getAge());
 return "one";
}

如何在枚举类中使用@Autowired

为什么要在枚举类中使用@Autowired,是因为我在做需求时将责任链设计模式和策略模式结合起来做了,关于策略模式不明白的话可以看我之前的文章设计模式——策略模式。我们可以使用枚举类使我们的代码更加清晰可见。

我们定义一个简单的枚举策略模式的枚举类。例如我们在这里面要使用Spring容器中得了类的话,我们该如何写呢?例如还是User类。

public enum HumanEnum {

 MAN("man"){
 @Override
 public void invoke() {
 log.info("i am man");
 }
 },
 WOMAN("woman"){
 @Override
 public void invoke() {
 log.info("i am woman");
 }
 };
 
 String value;

 HumanEnum(String value){
 this.value = value;
 }

 public abstract void invoke();
}

只需要在枚举类中定义一个内部类,然后将注入进来的类赋值给枚举类即可。

User user;

public void setUse(User user){
 this.user = user;
}

@Component
public static class HumanEnumInjector{

 @Autowired
 private User user;

 @PostConstruct
 public void setValue(){
 for (HumanEnum humanEnum : EnumSet.allOf(HumanEnum.class)){
 humanEnum.setUse(user);
 }
 }
}

总结

面向对象的三大特性:封装、继承、多态
  1. 大部分的设计模式都是围绕着面向对象的三大特性进行演化的
  2. @Autowired也可以定义在方法上,之前只是习惯将其定义在字段上
  3. 注重基础,复杂的设计肯定也是通过一个一个简单的设计将其拼合的
  • 1
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值