您现在的位置是:亿华云 > 系统运维
SpringBoot 自动装配的原理分析
亿华云2025-10-04 00:39:04【系统运维】7人已围观
简介自动装配案例首先我们通过一个案例来看一下自动装配的效果,创建一个SpringBoot的项目,在pom文件中加入下面的依赖。org.springframework.bootspring-boot-st
首先我们通过一个案例来看一下自动装配的自动装配效果,创建一个 SpringBoot 的理分项目,在 pom 文件中加入下面的自动装配依赖。
spring-boot-starter-web
spring-boot-starter-data-redis
</dependency>其中 web 的理分依赖表示我们这是一个 web 项目,redis 的自动装配依赖就是我们这边是要验证的功能依赖。随后在 application.properties 配置文件中增加 redis 的理分相关配置如下
spring.redis.host=localhost
spring.redis.port=6379
spring.redis.password=123456再编写一个 Controller 和 Service 类,相关代码如下。自动装配
package com.example.demo.controller;
import com.example.demo.service.HelloService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestParam;
import org.springframework.web.bind.annotation.RestController;
@RestController
public class HelloController {
@Autowired
private HelloService helloService;
@GetMapping(value = "/hello")
public String hello(@RequestParam("name") String name) {
return helloService.sayHello(name);
}
}service 代码如下
package com.example.demo.service;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.stereotype.Service;
@Service
public class HelloService {
@Autowired
RedisTemplate
public String sayHello(String name) {
String result = doSomething(name);
redisTemplate.opsForValue().set("name", result);
result = redisTemplate.opsForValue().get("name");
return "hello: " + result;
}
private String doSomething(String name) {
return name + " 欢迎关注 Java 极客技术";
}
}启动项目,然后我们通过访问 http://127.0.0.1:8080/hello?自动装配name=ziyou,可以看到正常访问。理分接下来我们再通过 Redis 的自动装配客户端,去观察一下数据是理分否正确的写入到 Redis 中,效果跟我们想象的自动装配一致。
看到这里很多小伙伴就会说,理分这个写法我天天都在使用,自动装配用起来是真的爽。虽然用起来是很爽,网站模板但是大家有没有想过一个问题,那就是在我们的 HelloService 中通过 @Autowired 注入了一个 RedisTemplate 类,但是我们的代码中并没有写过这个类,也没有使用类似于@RestController,@Service 这样的注解将 RedisTemplate 注入到 Spring IoC 容器中,那为什么我们就可以通过 @Autowired 注解从 IoC 容器中获取到 RedisTemplate 这个类呢?这里就是常说的自动装配的功能了。
首先我们看下项目的启动类;
package com.example.demo;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.EnableAutoConfiguration;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.context.annotation.ComponentScan;
@SpringBootApplication
@ComponentScan(value = "com.example.demo.*")
public class DemoApplication {
public static void main(String[] args) {
SpringApplication.run(DemoApplication.class, args);
}
}在启动类上面有一个 @SpringBootApplication 注解,我们点进去可以看到如下内容:
@Target({ ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Inherited
@SpringBootConfiguration
@EnableAutoConfiguration
@ComponentScan(
excludeFilters = { @Filter(
type = FilterType.CUSTOM,
classes = { TypeExcludeFilter.class}
), @Filter(
type = FilterType.CUSTOM,
classes = { AutoConfigurationExcludeFilter.class}
)}
)
public @interface SpringBootApplication {
// 省略
}在这个注解中,其中有一个 @EnableAutoConfiguration 注解,正是因为有了这样一个注解,我们才得以实现自动装配的功能。继续往下面看。
@Target({ ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Inherited
@AutoConfigurationPackage
@Import({ AutoConfigurationImportSelector.class})
public @interface EnableAutoConfiguration {
String ENABLED_OVERRIDE_PROPERTY = "spring.boot.enableautoconfiguration";
Class [] exclude() default { };
String[] excludeName() default { };
}
可以看到 @EnableAutoConfiguration 注解中有一个 @Import({ AutoConfigurationImportSelector.class}),导入了一个 AutoConfigurationImportSelector 类,该类间接实现了 ImportSelector 接口,实现了一个 String[] selectImports(AnnotationMetadata importingClassMetadata); 方法,这个方法的返回值是一个字符串数组,对应的是站群服务器一系列主要注入到 Spring IoC 容器中的类名。当在 @Import 中导入一个 ImportSelector 的实现类之后,会把该实现类中返回的 Class 名称都装载到 IoC 容器中。
一旦被装载到 IoC 容器中过后,我们在后续就可以通过 @Autowired 来进行使用了。接下来我们看下 selectImports 方法里面的实现,当中引用了 getCandidateConfigurations 方法 ,其中的 ImportCandidates.load 方法我们可以看到是通过加载 String location = String.format("META-INF/spring/%s.imports", annotation.getName()); 对应路径下的 org.springframework.boot.autoconfigure.AutoConfiguration.imports 文件,其中就包含了很多自动装配的配置类。
protected List
List
ImportCandidates.load(AutoConfiguration.class, this.getBeanClassLoader()).forEach(configurations::add);
Assert.notEmpty(configurations, "No auto configuration classes found in META-INF/spring.factories nor in META-INF/spring/org.springframework.boot.autoconfigure.AutoConfiguration.imports. If you are using a custom packaging, make sure that file is correct.");
return configurations;
}我们可以看到这个文件中有一个 RedisAutoConfiguration 配置类,在这个配置中就有我们需要的 RedisTemplate 类的 Bean,同时也可以看到,在类上面有一个 @ConditionalOnClass({ RedisOperations.class}) 注解,表示只要在类路径上有 RedisOperations.class 这个类的时候才会进行实例化。这也就是为什么只要我们添加了依赖,就可以自动装配的原因。
通过 org.springframework.boot.autoconfigure.AutoConfiguration.imports 这个文件,我们可以看到有很多官方帮我们实现好了配置类,这些功能只要我们在 pom 文件中添加对应的 starter 依赖,服务器租用然后做一些简单的配置就可以直接使用。
其中本质上自动装配的原理很简单,本质上都需要实现一个配置类,只不过这个配置类是官方帮我们创建好了,再加了一些条件类注解,让对应的实例化只发生类类路径存在某些类的时候才会触发。这个配置类跟我们平常自己通过 JavaConfig 形式编写的配置类没有本质的区别。
自动装配总结从上面的分析我们就可以看的出来,之所以很多时候我们使用 SpringBoot 是如此的简单,全都是依赖约定优于配置的思想,很多复杂的逻辑,在框架底层都帮我们做了默认的实现。虽然用起来很爽,但是很多时候会让程序员不懂原理,我们需要做的不仅是会使用,而更要知道底层的逻辑,才能走的更远。
基于上面的分析,我们还可以知道,如果我们要实现一个自己的 starter 其实也很简单,只要安装上面的约定,编写我们自己的配置类和配置文件即可。后面的文章阿粉会带你手写一个自己的 starter 来具体实现一下。
很赞哦!(87)
相关文章
- 二、如何选择合适的域名
- 脊叶数据中心架构有哪些优势?
- 大模型时代的算力革新,西云算力的三位一体战略
- Arm面向汽车行业推出多款新产品,将人工智能汽车开发周期缩短多达2年时间
- 5. 四种状态过后,域名管理机构释放域名给公众注册。
- 联想正式发布三大服务器新品,从云到端加速布局IT基础设施
- 浪潮云洲工业互联网平台V6.0正式发布
- 探访光子1号金融算力中心,北京最近的万座机架规模智算中心
- 用户邮箱的静态密码可能已被钓鱼和同一密码泄露。在没有收到安全警报的情况下,用户在适当的时间内不能更改密码。在此期间,攻击者可以随意输入帐户。启用辅助身份验证后,如果攻击者无法获取移动电话动态密码,他将无法进行身份验证。这样,除非用户的电子邮件密码和手机同时被盗,否则攻击者很难破解用户的邮箱。
- 负载均衡速成,你学会了吗?