SpringBoot自动配置(容器功能)
容器功能
SpringBoot的底层注解是怎样完成某些功能的。
1.1组件添加
@Configuration
/**
* @Configuration注解告诉SpringBoot是一个配置类 === 配置文件,
* 配置类里面使用@Bean注解标注方法上给容器注册组件,默认也是单实例的,
* 那么MyConfig也是一个组件,因为他是被@Configuration标注的
* proxyBeanMethods:代理bean的方法
* ProxyBeanMethods当问true时,表示外部无论对配置类中的Bean组件注册方法调用多少次获取的都是之前注册容器中的单实例对象
* ProxyBeanMethods当问true时,代理对象调用方法SpringBoot总会检查这个,保持组件单实例
* ProxyBeanMethods当问false时,调用bean组件注册方法获取的对象就不在相同,不在容器中找组件,每次都会产生一个新的对象
* 引出SpringBoot底层Configuration的两种配置
* Full(全局) proxyBeanMethods = true
* lite(轻量级) proxyBeanMethods = false
* 解决的场景就是组件依赖
*/
@Configuration(proxyBeanMethods = false)
public class MyConfig {
/**
* 在我们学习spring的时候,springIOC,创建对象和依赖注入我们是通过bean标签来实现的,在SpringBoot中,因为没有了Spring的配置文件
* 那么我们就可以使用@Bean注解充当bean标签
* @Bean 相当于 bean标签
* 返回值类型 相当于bean标签当中的class属性
* 方法名 相当于bean标签中的id
* 返回值 相当于bean标签中的property
* @Bean 给容器中添加组件,以方法名作为组件的id,返回值类型是组件的类型,返回的值,就是组件在容器中的实例
* @return
*/
@Bean
public User user01(){
return new User("wsj",22);
}
@Bean("bob")//加个括号相当于给id起了个别名,我们可以把组件名就不是方法名了变成别名
public Pet bobPet(){
return new Pet("bob");
}
}
@SpringBootApplication
public class MainApplication {
public static void main(String[] args) {
//返回IOC容器
ConfigurableApplicationContext run = SpringApplication.run(MainApplication.class, args);
//查看容器中的组件
String[] names = run.getBeanDefinitionNames();
for (String s:
names) {
System.out.println(s);
}
//获取容器中的组件
Pet bob1 = run.getBean("bob", Pet.class);
Pet bob2 = run.getBean("bob", Pet.class);
System.out.println(bob1==bob2);
MyConfig bean = run.getBean(MyConfig.class);
System.out.println(bean);
Pet pet = bean.bobPet();
Pet pet1 = bean.bobPet();
System.out.println(pet == pet1);
}
}
@Import
@Import({User.clss,DBHelper.class}) //{}中的类型可以任意写
//给容器中自动创建出这两个类型的组件,默认组件的名字就是全类名
//@Import注解用在任何组件的上边,像上面的@Configuration,或@Component,@Service都可以
@Conditional
条件装配,满足Conditional指定的条件,则进行组件注入
@Conditional有很多派生注解
@ConditionalOnBean当容器中存在某个Bean组件的时候我们可以干什么(注入什么组件)
@ConditionalOnMissingBean当容器中不存在某个Bean组件的时候我们可以干什么
@ConditionalOnClass当容器中存在某个Class的时候我们可以干什么
@ConditionalOnMissingClass当容器中不存在某个Bean组件的时候我们可以干什么
@ConditionalOnResource当项目的类路径里面存在某个资源的时候我们干什么
@ConditionalOnJava当Java版本号是什么的时候我们干什么
@ConditionalOnWebApplication当应用是一个Web应用的时候我们干什么
@ConditionalOnNotWebApplication当应用不是Web应用的时候我们干什么
@ConditionalOnBean(name = "bob2")//如果存在bob2组件则装配user01组件,否则则不装配
@Bean
public User user01(){
return new User("wsj",22);
}
@ConditionalOnBean(name = "bob2")
//如果存在bob2组件则装配类中全部组件,否则不装配类中全部组件
public class MyConfig {
@Bean
public User user01(){
return new User("wsj",22);
}
@Bean("bob")
public Pet bobPet(){
return new Pet("bob");
}
}
@ImportResource
导入资源
如果我们是使用原生的bean标签的形式来进行的组件定义,我们又不想一步步的把bean标签形式转换成@Bean注解形式,那么我们就可以使用@ImportResource来进行
@ImportResource放到其他组件中,如@Configuration,@Service,@Component等
@ImportResource("classpath:bean.xml")//bean.xml是自己定义的bean标签形式文件让其生效
@ConfigurationProperties
如何使用Java读取到properties文件中的内容,并且把它封装到JavaBean中,以供随时使用。(就是我们读取配置文件中关于bean属性的值赋值给bean,我们使用mysql时,配置文件的地址,用户名,密码这些,我们都可以拥有一个拥有这些属性的对象,然后使用@ConfigurationProperties来进行赋值)
只有容器中的组件才能享有SpringBoot的强大功能
第一种方式
//只有容器中的组件才可以享有它的强大功能
@Component
@ConfigurationProperties(prefix = "car")
public class Car {
private String brand;
private Integer price;
public Car() {
}
public String getBrand() {
return brand;
}
public void setBrand(String brand) {
this.brand = brand;
}
public Integer getPrice() {
return price;
}
public void setPrice(Integer price) {
this.price = price;
}
@Override
public String toString() {
return "Car{" +
"brand='" + brand + '\'' +
", price=" + price +
'}';
}
}
car.brand = YD
car.price = 100000
第二种方式@ConfigurationProperties+@EnalbeConfigurationProperties
@Configuration//必须在配置类中才能写这个注解
@EnableConfigurationProperties(Car.class)
//我们就不需要在Car类上在写一个@Component注解了,但是@ConfigurationProperties还是要写的
//@EnableConfigurationProperties有两个功能
//1.开启Car配置绑定功能
//2.把这个Car这个组件自动注册到容器中