JavaEE开发之Spring中的条件注解、组合注解与元注解

上篇博客我们详细的聊了《JavaEE开发之Spring中的多线程编程以及任务定时器详解》,本篇博客我们就来聊聊条件注解@Conditional以及组合条件。条件注解说简单点就是根据特定的条件来选择Bean对象的创建。条件注解就是可以根据不同的条件来做出不同的事情。在Spring中条件注解可以说是设计模式中状态模式的一种体现方式,同时也是面向对象编程中多态的应用部分。而组合注解就是将现有的注解进行组合。下方会给出具体的介绍和实例。

一、条件注解----@Conditional

本篇博客的本部分我们来聊一下条件注解,顾名思义,条件注解就是可以根据不同的条件来做出不同的事情。在Spring中条件注解可以说是设计模式中状态模式的一种体现方式,同时也是面向对象编程中多态的应用部分。

在Spring框架中,当我们使用条件注解时,我们会为每种独立的条件创建一个类,根据这个类对应的条件的成立情况我们来选择不同的任务来执行。当然我们在声明任务时,一般使用接口来声明。因为我们会在Spring的配置类中指定具体条件下的具体类。接下来,我们将来看一下Spring框架中@Conditional注解的具体使用方式。

当然同一个Service接口所对应的条件集合中是互斥的,也就是说在特定情况下只有一个条件成立。

1、创建服务接口以及具体的服务类

首先我们来创建一个Service的接口,然后再基于遵循该接口的情况下来创建两个Service类。下方我们将会在配置类中指定不同条件下会对应不同的Service对象。首先我们先来创建Service的接口。下方这段代码就是我们创建的Service的接口,该接口比较简单,只有一个描述方法。在具体是Service类中我们将会给出description()方法的具体实现,用此方法来区分不同类的实现。

package com.zeluli.conditional;

public interface ConditinalServiceInteface {
    public String description();
}

创建完ServiceInterface后,我们就该创建具体的类了。下方的FirstConditionService和SecondConditionService两个类都实现了ConditinalServiceInteface接口,并且给出了description()方法的具体实现。稍后,我们将会在下方类配置Bean时,给出相应的条件。本小节只是准备部分。

package com.zeluli.conditional;

public class FirstConditionService implements ConditinalServiceInteface {
    public String description() {
        return "第一个条件成立的Service";
    }
}


=========================================

package com.zeluli.conditional;

public class SecondConditionService implements ConditinalServiceInteface {
    public String description() {
        return "第二个条件成立的Service";
    }
}

2、创建@Conditional对应的条件类

创建完Service接口以及Service类后,接下来我们就来创建@Conditional注解所需的条件类。每个条件类对应着一种独立的情况,在Spring中的条件类需要实现Condition接口。下方是我们创建的两个条件类。

这两个条件类都实现了Spring框架中的Condition,并且给出了matches()方法的实现。matches()方法的返回值是一个布尔类型的值,如果返回false说明该条件类所对应的条件不成立,如果返回true则说明该条件对应的条件成立。为了简化操作,我们就指定FirstConditional对应的条件为false,而SecondConditional对应的条件为true。具体的条件类的实现如下所示。

package com.zeluli.conditional;

import org.springframework.context.annotation.Condition;
import org.springframework.context.annotation.ConditionContext;
import org.springframework.core.type.AnnotatedTypeMetadata;

public class FirstConditional implements Condition {
    //提供条件的方法
    public boolean matches(ConditionContext context, AnnotatedTypeMetadata metadata) {
        return false;
    }
}

==========================================

package com.zeluli.conditional;

import org.springframework.context.annotation.Condition;
import org.springframework.context.annotation.ConditionContext;
import org.springframework.core.type.AnnotatedTypeMetadata;

public class SecondConditional implements Condition {
    public boolean matches(ConditionContext context, AnnotatedTypeMetadata metadata) {
        return true;
    }
}

3.在Java配置类中进行条件配置

Service的接口、Service的类以及相应的条件创建完毕后,接下来我们就该在Java的配置类中将条件类与Service类对象进行关联了。下方代码段就是该部分对应的配置类。在声明FirstConditionService类的Bean时,我们使用@Conditional注解,@Conditional的参数为FirstConditional.class,也就是说明当FirstConditional类所对应的条件成立时FirstConditionService的对象才会被实例化。

同理,下方的SecondConditionService对应的条件是SecondConditional。具体代码如下所示。

package com.zeluli.conditional;

import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Conditional;
import org.springframework.context.annotation.Configuration;

@Configuration
@ComponentScan("com.zeluli.conditional")
public class ConditionalConfig {
    
    @Bean
    @Conditional(FirstConditional.class) //指定条件类
    public FirstConditionService getFirstConditionalService() {
        return new FirstConditionService();
    }
    
    @Bean
    @Conditional(SecondConditional.class)
    public SecondConditionService getSecondConditionService() {
        return new SecondConditionService();
    }
}

4、创建Main方法进行测试

接下来又到了测试的时刻了,下方我们从上面的ConditionalConfig配置类中获取上下文,然后从上下文中获取相应的Service对象。在获取对象时,我们使用的是ConditionalServiceInterface接口来获取和声明的Bean。也就是说service变量所承载的对象是实现ConditionalServiceInterface接口的所有类中的某个类的对象。当某个Service类所对应的条件成立时,该类的对象就会被创建。

上面我们也提到过,ConditionalServiceInterface接口下每个类对应的这些条件必须是互斥的,也就是这些条件在特定情况下只有一个是成立的。因为我们为第二个条件返回的是true, 所以该条件是成立的,那么SecondConditionalService类的对象就会被调用。所以我们调用service的description()方法时,调用的是SecondConditionalService类中的相应的方法。具体如下所示。

二、组合注解

组合注解这个就比较好理解了,就是将多个注解组合到一块生成一个新的注解。使用这个新的注解就相当于使用了该组合注解中所有的注解。这个特性还是蛮有用的,接下来我们就来看一下如何创建和使用组合注解。

1.组合注解的创建

接下来我们就通过一个简单的实例来看一下如何将多个注解组合到一块。在之前的Spring配置类中,我们经常使用到@Configuration和@ComponentScan这两个注解,接下来,我们将其进行组合封装,从而形成一个新的注解。

下方这个CombinationConfiguration注解就是我们组合的新的注解,该注解中使用了@Configuration和@ComponentScan进行修饰,也就说明@CombinationConfiguration注解兼有@Configuration和@ComponentScan这两个注解的功能。

package com.zeluli.combination.annotation;

import java.lang.annotation.Documented;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;

@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Configuration
@ComponentScan
public @interface CombinationConfiguration {
    String[] value() default{};
}

2.组合注解的使用

创建完相应的组合注解后就到了使用的时候了,上面注解的使用和一般的注解没有什么区别。只是这个注解表示之前写的@Configuration和@ComponentScan这两个注解。下方代码截图就是该组合注解的使用方式。

OK,今天博客就先到这儿吧,github源码分享地址:https://github.com/lizelu/SpringDemo

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏Java3y

JSP第三篇【JavaBean的介绍、JSP的行为--JavaBean】

什么是javaBean JavaBean就是一个普通的java类,也称之为简单java对象--POJO(Plain Ordinary Java Object),...

39560
来自专栏IT技术精选文摘

跟着实例学习ZooKeeper的用法: 缓存

可以利用ZooKeeper在集群的各个节点之间缓存数据。 每个节点都可以得到最新的缓存的数据。 Curator提供了三种类型的缓存方式:Path Cache,N...

32570
来自专栏陈本布衣

JAXB应用实例

过往的项目中数据存储都离不开数据库,不过最近做的一个项目的某些数据(比如人员信息、菜单、权限等等)却完全没有涉及任何数据库操作,直接XML搞定。这里无意比较优...

42690
来自专栏张善友的专栏

怎么解决web service circular reference 问题

下面是一个非常简单的例子: using System; using System.Data; using System.Configuration; using...

22150
来自专栏木可大大

初始Java

为了让Java能够实现跨平台,Java的发明者们增加了一个抽象层,即JVM(Java Virtual Machine,Java虚拟机),自定义一套指令并且和硬件...

9320
来自专栏阿杜的世界

Spring Boot:定制PropertyEditors

在Spring Boot: 定制HTTP消息转换器一文中我们学习了如何配置消息转换器用于HTTP请求和响应数据,实际上,在一次请求的完成过程中还发生了其他的转换...

8740
来自专栏黑泽君的专栏

java中的类型安全问题-Type safety: Unchecked cast from Object to ... 或者 Type safety: Unchecked cast from Type

首先,java语言室类型安全的,通常我们遇到这个问题是出现在 Object转化为目标类型 或者 Type转化为目标类型 时,

23820
来自专栏函数式编程语言及工具

泛函编程(35)-泛函Stream IO:IO处理过程-IO Process

    IO处理可以说是计算机技术的核心。不是吗?使用计算机的目的就是希望它对输入数据进行运算后向我们输出计算结果。所谓Stream IO简单来说就是对一串按序...

212100
来自专栏圣杰的专栏

线程安全知多少

1. 如何定义线程安全 线程安全,拆开来看: 线程:指多线程的应用场景下。 安全:指数据安全。 多线程就不用过多介绍了,相关类型集中在System.Thread...

35350
来自专栏用户画像

Spring中bean的作用域

1、singleton作用域 当一个bean的作用域设置为singleton, Spring IOC容器只会创建该bean定义...

6920

扫码关注云+社区

领取腾讯云代金券