学习SpringBoot容器功能及注解原理
作者:LL.LEBRON 发布时间:2023-11-24 22:06:17
1.组件添加
1.1@Configuration
@Configuration:告诉SpringBoot这是一个配置类
配置类里面使用@Bean
标注在方法上给容器注册组件,默认也是单实例的
配置类本身也是组件
proxyBeanMethods
:代理bean的方法
Full(
proxyBeanMethods = true
):保证每个@Bean方法被调用多少次返回的组件都是单实例的Lite(
proxyBeanMethods = false
):每个@Bean方法被调用多少次返回的组件都是新创建的组件依赖必须使用Full模式默认。其他默认是否Lite模式
最佳实战:
1.配置类组件之间无依赖关系用Lite模式加速容器启动过程,减少判断
2.配置类组件之间有依赖关系,方法会被调用得到之前单实例组件,用Full模式
代码实战演示:
@Configuration(proxyBeanMethods = false)//告诉SpringBoot这是一个配置类=配置文件
public class MyConfig {
@Bean//给容器中添加组件。以方法名作为组件的id。返回类型就是组件类型。返回的值,就是组件在容器中的实例
public User user01() {
User zhangsan = new User("zhangsan", 18);
return zhangsan;
}
@Bean("tom")//也可以自己设置id代替方法名作为id
public Pet tomcatPet() {
return new Pet("tomcat");
}
}
@SpringBootApplication
public class Boot01HelloworldApplication {
public static void main(String[] args) {
ConfigurableApplicationContext run = SpringApplication.run(Boot01HelloworldApplication.class, args);
MyConfig bean = run.getBean(MyConfig.class);
System.out.println(bean);//com.atguigu.boot.config.MyConfig@d67d8
//如果@Configuration(proxyBeanMethods = true)代理对象调用方法。SpringBoot总会检查这个组件是否在容器中有。
//保持组件单实例
User user = bean.user01();
User user1 = bean.user01();
//(proxyBeanMethods = true)返回true
//(proxyBeanMethods = false)返回false
System.out.println(user == user1);
}
}
如果有组件依赖:
@Configuration(proxyBeanMethods = true)//告诉SpringBoot这是一个配置类=配置文件
public class MyConfig {
@Bean//给容器中添加组件。以方法名作为组件的id。返回类型就是组件类型。返回的值,就是组件在容器中的实例
public User user01() {
User zhangsan = new User("zhangsan", 18);
//user组件依赖了Pet组件
zhangsan.setPet(tomcatPet());
return zhangsan;
}
@Bean("tom")//也可以自己设置id代替方法名作为id
public Pet tomcatPet() {
return new Pet("tomcat");
}
}
@SpringBootApplication
public class Boot01HelloworldApplication {
public static void main(String[] args) {
ConfigurableApplicationContext run = SpringApplication.run(Boot01HelloworldApplication.class, args);
MyConfig bean = run.getBean(MyConfig.class);
System.out.println(bean);
User user01 = run.getBean("user01", User.class);
Pet tom = run.getBean("tom", Pet.class);
//(proxyBeanMethods = true)返回(用户的宠物:true)
//(proxyBeanMethods = false)返回(用户的宠物:false)
System.out.println("用户的宠物:"+(user01.getPet() == tom));
}
}
1.2@Import
@Import:给容器中导入组件
代码演示:
//给容器中自动无参构造创建出这两个类型的组件、默认组件的名字就是全类名
@Import({User.class, DBHelper.class})
@Configuration(proxyBeanMethods = false) //告诉SpringBoot这是一个配置类 == 配置文件
public class MyConfig {
}
@SpringBootApplication
public class Boot01HelloworldApplication {
public static void main(String[] args) {
ConfigurableApplicationContext run = SpringApplication.run(Boot01HelloworldApplication.class, args);
//获取组件
String[] beanNamesForType = run.getBeanNamesForType(User.class);
for (String s : beanNamesForType) {
System.out.println(s);
}
DBHelper bean = run.getBean(DBHelper.class);
System.out.println(bean);
}
}
//输出:
com.atguigu.boot.bean.User
ch.qos.logback.core.db.DBHelper@16ef799
1.3@Conditional
@Conditional:条件装配,满足Conditional
指定的条件,则进行组件注入
有一系列派生注解:
2.原生配置文件引入
2.1@ImportResource
原生xml文件:
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:context="http://www.springframework.org/schema/context"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context https://www.springframework.org/schema/context/spring-context.xsd">
<bean id="haha" class="com.atguigu.boot.bean.User">
<property name="name" value="zhangsan"></property>
<property name="age" value="18"></property>
</bean>
<bean id="hehe" class="com.atguigu.boot.bean.Pet">
<property name="name" value="tomcat"></property>
</bean>
</beans>
自定义配置类:
@Configuration(proxyBeanMethods = true)//告诉SpringBoot这是一个配置类=配置文件
@ImportResource("classpath:beans.xml")
public class MyConfig {
}
测试:
@SpringBootApplication
public class Boot01HelloworldApplication {
public static void main(String[] args) {
ConfigurableApplicationContext run = SpringApplication.run(Boot01HelloworldApplication.class, args);
boolean haha = run.containsBean("haha");
boolean hehe = run.containsBean("hehe");
System.out.println(haha);//true
System.out.println(hehe);//true
}
}
3.配置绑定
如何使用Java读取到properties文件中的内容,并且把它封装到JavaBean中,以供随时使用;
原生方法(配置文件复杂就显得麻烦):
public class getProperties {
public static void main(String[] args) throws FileNotFoundException, IOException {
Properties pps = new Properties();
pps.load(new FileInputStream("a.properties"));
Enumeration enum1 = pps.propertyNames();//得到配置文件的名字
while(enum1.hasMoreElements()) {
String strKey = (String) enum1.nextElement();
String strValue = pps.getProperty(strKey);
System.out.println(strKey + "=" + strValue);
//封装到JavaBean。
}
}
}
3.1@ConfigurationProperties
配置文件:
mycar.brand=BYD
mycar.price=100000
创建一个car类:
//只有在容器中的组件,才会拥有SpringBoot提供的强大功能
@Component
@ConfigurationProperties(prefix = "mycar")
//Lombok注解简化开发
@Data
@NoArgsConstructor
@ToString
@AllArgsConstructor
public class Car {
private String brand;
private Integer price;
}
测试方法:
@RestController
public class HelloController {
@Autowired
Car car;
@RequestMapping("/car")
public Car car(){
return car;
}
}
测试结果:
3.2@EnableConfigurationProperties + @ConfigurationProperties
@EnableConfigurationProperties
必须在配置类里写:
@Configuration(proxyBeanMethods = true)//告诉SpringBoot这是一个配置类=配置文件
@EnableConfigurationProperties(Car.class)
//1.开启Car属性配置绑定功能
//2.把Car这个组件自动注册到容器中
public class MyConfig {
}
该写法就不用在写@Component
@ConfigurationProperties(prefix = "mycar")
@Data
@NoArgsConstructor
@ToString
@AllArgsConstructor
public class Car {
private String brand;
private Integer price;
}
来源:https://blog.csdn.net/qq_45966440/article/details/120412878
猜你喜欢
- log4j2支持日志的异步打印,日志异步输出的好处在于,使用单独的进程来执行日志打印的功能,可以提高日志执行效率,减少日志功能对正常业务的影
- mybatis-plus自动配置mapper.xml与java接口映射本来没有mybatis-plus的话,这个工作是通过mybatis-s
- 单行文本的输入存在严重的缺陷,也不适合实际的运用,本节通过一个无功能的记事本来介绍可以进行多行输入的JTextArea:JTextArea(
- 本文实例为大家分享了java简单实现斗地主发牌的具体代码,供大家参考,具体内容如下问题:参考斗地主的游戏规则,完成一个发牌的功能(54张牌,
- 代码如下:try { // 创建一个线程 Thread thread = new Thread() {
- 关于idea2021最新激活教程,请点击此处,获取最新激活教程还有一种激活方法,点击此处获取吧 !下面看下IDEA 2021.2 启动报错问
- 定义与结构 备忘录(Memento)模式又称标记(Token)模式。GOF给备忘录模式的定义为:在不破坏
- Java xml出现错误 javax.xml.transform.TransformerException: java.lang.NullP
- 初学线程时,总是将 run 方法和 start 方法搞混,虽然二者是完全不同的两个方法,但刚开始使用时很难分清,原因就是因为初次使用时效果貌
- Java项目涉及到数据库交互,以往常用的是JDBC,现在则有Hibernate、Mybatis等这些持久化支持。项目中用到了MyBatis,
- 本文主要和大家分享介绍了关于Java JDK * 使用的相关内容,分享出来供大家参考学习,下面来一起看看详细的介绍:前言代理是一种常用的
- 1.ArrayList 是基数组结构的,需要连续的内存空间从构造函数可以看出,ArrayList内部用一个Object数组来保存数据。对于无
- 1.概述数据库开发一直是JAVA开发的核心之一,作为现在JAVA EE的基石框架,Spring Boot自身携带了一个JDBCTemplat
- 双向顺序队列ArrayDeque和双向链式队列LinkedList,JDK已经包含,在此略。ArrayDeque包括顺序栈和顺序队列,Lin
- 一、数组创建1.1 声明并赋值int[] a = {1,2,3};1.2 声明数组名开辟空间并且赋值int[] a;a = new int[
- 前言空间分配要点有:一是空间分配的连续性;二是动态内存申请;三是防止程序执行中出现异常错误。提示:开始讲解了嗷~后续会根据精力持续更新嗷!!
- java项目中常用maven工具来进行工程管理,但经常遇到的一个问题是生成的jar包越来越大,编译一次工程越来越慢。怎么有效地去除冗余依赖,
- — 遇到问题今天在IDEA里面运行项目的时候报了一个错,如下图所示:— 找到问题根源其实控制台给出的错误信息提示说的很明显:类加载器加载文件
- 获取和释放 monitor 锁的时机本文我们研究下 synchronized 背后的 monitor 锁。我们都知道,最简单的同步方式就是利
- 这一篇我们说说Java线程Thread的interrupt中断机制。中断线程线程的thread.interrupt()方法是中断线程,将会设