目录
前言
1. 自定义condition加载bean
1.1. 自定义一个condition注解
1.2. 实现自定义注解对应的实现类
1.3. 使用如上注解
1.4. 使用Spring上下文获取一下改bean
2. 我们来看看Spring是如何加载redisTemplate的。
2.1. 找到Spring的autoconfigure的jar包,我们从中可以看到有很多对应的condition注解及他们的实现类:编辑
2.2. 要初始化出来redisTemplate出来有很多条件,其中一个就是要有redis对应的依赖中的字节码文件。编辑
3. Spring自动切换Web服务器(Tomcat/Netty/Jetty ...)
总结Spring Condition
前言
我们都知道,当我们创建一个基于Spring的项目的时候,我们都需要在Applicaiton启动引导类上加上@ComponentScan("com.acom.springdemo.*")注解,用于引导Spring指定扫描范围。如果我们不写的话Spring默认扫描的就只有Application启动类所在的根目录及其子目录。Spring为什么呢一定要指定扫描范围。试想一下我们有一个大项目,它依赖了很多的jar包,拿到要Spring启动时都将他们扫描一次,Spring又如何知道当前项目有哪些jar包,如果把所有的jar都扫描一次,很多jar是没有bean的,是不是会引起Spring项目启动很慢等问题。
有的时候我们需要引用外部jar包如Redis,显然这个外部jar包不在我们@ComponentScan所指定的范围内,那么Spring是如何自动加载它的呢?今天我们就来学些Spring condition在其中的作用。
1. 自定义condition加载bean
假设我们有这样一个需求,我们自定义一个user类的bean,但是这个bean需要在我们的pom中引用了Jedis和FastJson这两个jar包之后才能加载该bean,否则不能加载。
因此我们就需要自定义一个codition 用于限制bean的加载。
自定义bean的示意图如下:
1.1. 自定义一个condition注解
package com.mycompany.condition;import org.springframework.context.annotation.Conditional;
import java.lang.annotation.*;@Target({ElementType.TYPE, ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Conditional(ClassConditionImpl.class)
public @interface MyConditionOnClass {String[] value();
}
自定义一个condition注解类MyConditionOnClass,其中@Conditional(ClassConditionImpl.class)是说明当前注解是一个Condition注解,且该注解的具体实现类是ClassConditionImpl.class。这里我们需要明白的是,注解的本质其实就是一个标记,标记在方法或类上等,标记了之后它的实现类才能知道哪些方法需要做对应的处理,因而注解本身是不包含什么逻辑的,它只是一个标记。
1.2. 实现自定义注解对应的实现类
public class ClassConditionImpl implements Condition {@Overridepublic boolean matches(ConditionContext context, AnnotatedTypeMetadata metadata) {Map<String, Object> map = metadata.getAnnotationAttributes(MyConditionOnClass.class.getName());String[] value = (String[]) map.get("value");try {for (String className : value){Class<?> cls = Class.forName(className);}return true;}catch (ClassNotFoundException ex){return false;}}
}
自定义注解的实现类必须实现Spring的 Condition接口,因为Spring需要通过该接口传入Spring的上下文,以及自定义注解定义的参数信息,以便我们的实现类使用。详细如下:
- ConditionContext context:
上下文对象。用于获取环境,IOC容器,ClassLoader对象等。 - AnnotatedTypeMetadata metadata
注解的元对象,可以用于获取注解定义的元属性值
1.3. 使用如上注解
@Configuration
public class UserConfig {@Bean@MyConditionOnClass({"redis.clients.jedis.Jedis","com.alibaba.fastjson.JSON"})public User user(){return new User();}
}
1.4. 使用Spring上下文获取一下改bean
@SpringBootApplication
@ComponentScan
public class SpringbootConditionApplication {public static void main(String[] args) {ConfigurableApplicationContext context = SpringApplication.run(SpringbootConditionApplication.class,args);Object user = context.getBean("user");System.out.println(user);}
}
测试当我们没有引用如上的jar包时:
Exception in thread "main" org.springframework.beans.factory.NoSuchBeanDefinitionException: No bean named 'user' available
at org.springframework.beans.factory.support.DefaultListableBeanFactory.getBeanDefinition(DefaultListableBeanFactory.java:874)
当我们引用了如上的jar包时:
com.mycompany.domain.User@7c71c889
2. 我们来看看Spring是如何加载redisTemplate的。
2.1. 找到Spring的autoconfigure的jar包,我们从中可以看到有很多对应的condition注解及他们的实现类:
2.2. 要初始化出来redisTemplate出来有很多条件,其中一个就是要有redis对应的依赖中的字节码文件。
3. Spring自动切换Web服务器(Tomcat/Netty/Jetty ...)
我们都知道基于Spring的项目默认使用的是Tomcat web服务器。那么我们如何切换为其他web服务器呢?我们可以看到对应服务器的加载也是基于@ConditionalOnClass注解的。
所以如果我们要不使用Tomcat而是要使用其他的web服务器,我们只需要排除Tomcat然后再加载对应的web服务器依赖就可以了
<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-web</artifactId><exclusions><exclusion><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-tomcat</artifactId></exclusion></exclusions></dependency><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-jetty</artifactId></dependency>
启动日志
2024-06-09 18:06:58.223 INFO 15600 --- [ main] o.s.b.web.embedded.jetty.JettyWebServer : Jetty started on port(s) 8080 (http/1.1) with context path '/'
2024-06-09 18:06:58.231 INFO 15600 --- [ main] c.m.SpringbootConditionApplication : Started SpringbootConditionApplication in 3.385 seconds (JVM running for 3.972)
总结Spring Condition
Spring Condition的作用主要是根据特定条件来控制Bean的创建和注册行为。这种机制使得Spring框架在进行依赖注入时能够更加灵活和智能化。具体来说,其作用可以归纳为以下几点:
- 条件化Bean的注册:通过@Conditional注解,可以指定Bean注册到Spring IOC容器中的条件。只有当这些条件满足时,相应的Bean才会被创建和注册。这种动态注册Bean的方式提高了Spring应用的灵活性和可配置性。
- 实现自动化配置:Spring Condition机制常用于Spring Boot等框架中,以实现自动化配置。根据应用环境的不同,例如开发环境、测试环境和生产环境,可以自动地选择性地注册Bean,从而简化了配置工作。
- 整合环境和配置:通过ConditionContext对象,可以获取到当前的环境信息,如操作系统类型、Java版本等。这使得我们可以根据这些环境信息来动态地决定是否注册某个Bean。
- 扩展性和灵活性:由于Condition是一个函数式接口,开发者可以自定义实现类来重写matches()方法,以定义自己的条件判断逻辑。这种设计使得Spring Condition具有极高的扩展性和灵活性。
总的来说,Spring Condition的作用在于提供了一种根据特定条件动态注册Bean的机制,使得Spring框架在进行依赖注入时能够根据实际情况做出更智能的决策。这种机制在Spring Boot等框架中得到了广泛应用,极大地简化了应用的配置和部署工作。