![从企业级开发到云原生微服务:Spring Boot实战](https://wfqqreader-1252317822.image.myqcloud.com/cover/257/33831257/b_33831257.jpg)
3.2 Spring Bean的配置
3.2.1 注解配置(@Component)
当类注解为@Component、@Service、@Repository或@Controller时,Spring容器会自动扫描(通过@ComponentScan实现,Spring Boot已经做好了配置),并将它们注册成受容器管理的Bean。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_46_02.jpg?sign=1739615482-WVBqtp5VxjqL28DkveMb60Fe4hNjDeVS-0-dd98f2479aaf877a40d330c73ee12cb4)
@Component、@Service、@Repository和@Controller在当前示例中是完全等同的。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_46_03.jpg?sign=1739615482-jSyckLyKNxPdjlqKuILAjN0If30N5GWr-0-23e4252e622f89406fde3b6d24970e73)
上面的@Component和@Service都没有给Bean命名,Spring容器会自动命名为类名的第一个字母的小写形式,即someService和someService2。一般来说,没有必要去修改Bean的名称,使用默认的Bean名即可。当然,也可以通过@Component("SomeService")来设置Bean的名称。
@Service、@Repository和@Controller这三个注解组合了@Component注解,它们是@Component语义上的特例。
◎@Component:被注解类是“组件”。
◎@Controller:被注解类是“控制器”。
◎@Service:被注解类是“服务”。
◎@Repository:被注解类是“数据仓库”。
3.2.2 Java配置(@Configuration和@Bean)
在类上注解@Configuration(@Component的特例,会被容器自动扫描),可使类成为配置类。如果使用@Bean标注在类的方法上,则方法的返回值即为Bean的实例。假如现在有另外一个类。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_47_01.jpg?sign=1739615482-K3J8lxucpuRh6Wl1ouxKETZyOz2Wfsi0-0-83c609e5fe0f16fb55e193e3571444b2)
用Java配置的如下。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_47_02.jpg?sign=1739615482-M1d7qsCL7Ikasxe4Rik8OuGPpiy05i9f-0-b7c106f9c740aa0a05201a1796b46220)
同样,没有给Bean命名。Spring会将方法名anotherService默认成Bean的名称。若需要修改,则使用@Bean(name="AnotherService")。
3.2.3 依赖注入(Dependency Injection)
1.自动注入(@Autowired)
容器已经创建了SomeService、AnotherService和SomeService2的Bean,其他的Bean应如何注入使用呢?
(1)注解注入。
AnnotationInjectionService需要使用SomeService和AnotherService的Bean,我们只需在AnnotationInjectionService构造器上注解@Autowired,即可注入参数里需要的Bean。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_47_03.jpg?sign=1739615482-x45m3XBoJb0Tk8MSwhlTgdatzUTlisU5-0-3dc965e61f8473a6fe7459a26c2045cb)
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_48_01.jpg?sign=1739615482-6piv7pOo97cX89dZfREW4bFo6Mdnl5tQ-0-273248dd5e5ca81c7822dd8b466c971b)
在构造器上注解注入是Spring推荐的注入方式,当然,也可以通过在属性上注解@Autowired来注入Bean。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_48_02.jpg?sign=1739615482-ON4LwaJkwRM8H0Z1jZA50yb4UIiiPJJ6-0-171f50d6fe6ce69e836cd9e10e39326d)
还可以在set方法上注解@Autowired来注入Bean。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_48_03.jpg?sign=1739615482-qiA9l6g7fUGV1pKfq5gkDdfgmszfoKnB-0-bb936b9fa84a35fbbfee0a3194c19edb)
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_49_01.jpg?sign=1739615482-rxuSjIHmrBy8m8J4vjduuReVLYW7bood-0-5b4174da51ebb2f5fd83bf41ac2131a5)
如果Bean只有一个构造器,则可以直接省略@Autowired注解。若Bean有多个构造器,则需注解一个构造器用来注入,示例如下。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_49_02.jpg?sign=1739615482-2BAFkySoa8WMWPXXRx7ixKpfAQCzmMsm-0-599de194aa64eeb123cb59ab01347fdd)
(2)配置注入。
现在使用Java配置的方式在Bean JavaConfigInjectService中注入BeanAnotherService,JavaConfigInjectService定义如下。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_49_03.jpg?sign=1739615482-7ocaLs8XC1LPRz7teQbKRnujoaRLQHXN-0-8960c9aee5d05d50cad39a41c8098ecf)
前面已经将AnotherService通过@Bean注解成Bean了,下面只需在定义JavaConfigInjectService的Bean的方法参数里注入AnotherService的Bean即可。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_49_04.jpg?sign=1739615482-Nt7sKRmhZEq1FaOJkZnAGVSQlhqzalZz-0-417b6c1af1ce5dcde51d3057cb597c07)
在同一个配置类里,还可以在新建JavaConfigInjectService的构造里直接注入创建SomeService2的Bean的方法。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_49_05.jpg?sign=1739615482-DVxIqrNmpNhubtoKNiNlvJOha5WwKH6z-0-5423cbce21d8b3ec834322c14846f4d8)
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_50_01.jpg?sign=1739615482-addC0Q2eUFLq1K5GnqpTKJWmondUEcWN-0-ad23daa0f19d4fd56a91d96b1b753f15)
(3)混合注入。
注解配置的Bean可以直接注入给使用Java配置的Bean,反之亦然。
把注解配置的Bean注入Java配置的Bean:
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_50_02.jpg?sign=1739615482-khwobyMUPbuCBZGWqpKQ0hMj4e35iOa7-0-a42b612612f3b925990eb7bc6491c888)
把Java配置的Bean注入注解配置的Bean。被注入的BeanMixInjectionService2定义如下。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_50_03.jpg?sign=1739615482-GgIkZ8V4qIoFiLgOZHx9o87AJ5sJMlOR-0-af54bf89c96b441ddfe2dd6d45fe6c31)
在JavaConfig类里,可以直接在参数中注入Bean。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_50_04.jpg?sign=1739615482-swuE1OM2bHoAWFZMgHkv4iUc0ZqU20I7-0-b7f788cc1ead49023d8fce813fc3bdf0)
2.@Primary
上面的例子都是通过Bean的名称来自动注入的。当Bean的名称不满足条件时,容器会根据Bean的类型进行自动注入。当全局只有一个类型的Bean时,自动注入是没有问题的,但是当全局有多个同类型的Bean时,会提示“required a single bean, but n were found”,此时可以通过@Primary来注解需要优先使用的Bean。假如有两个Bean:
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_50_05.jpg?sign=1739615482-Jf5Hm7WqI1G48uRojWQeoDfL4jFAfvEC-0-88d1f504dbfe1fede2d978dfed3c9471)
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_51_01.jpg?sign=1739615482-RbAirh0BIrcIr1zP6wAxKZ7EIwWkPtn9-0-b11d4a47228b839d690b4a3c3eacd62f)
此时有两个Bean,名称分别为anotherService和primaryAnotherService。如果在注入的地方不使用这个两个名称,那么就会按照Bean的类型自动注入。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_51_02.jpg?sign=1739615482-yWJkIyA51gQ54NmjWcg5QZRVMZPI0iTN-0-7612f7e77992060e73f8e32213d3347b)
因为现在使用的 service不符合按照名称自动注入,所以是按照类型自动注入的。因为primaryAnotherService注解了@Primary,所以使用primaryAnotherService这个Bean。
3.@Qualifier
在上面的例子中,使用UsePrimaryService注入的AnotherService的Bean只能是primaryAnotherService,这时可以使用@Qualifier直接指定需要使用哪个Bean。还是使用上面例子中的两个Bean。
注入anotherService:
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_51_03.jpg?sign=1739615482-NGz9FYLclj4NWFvrudz3apsm4F5qEtsv-0-b958dc9d5deea3283ec2f4adf9c9021e)
注入primaryAnotherService。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_52_01.jpg?sign=1739615482-zScWl5bvDmhkzsIBzMWwrPiFA9ESSubK-0-d4bf570dd5ac3537b818d5b12f550f8b)
3.2.4 运行检验(CommandLineRunner)
在Spring Boot下可以注册一个CommandLineRunner的Bean,在容器启动后,这个Bean可用来执行一些专门的任务,如在JavaConfig里。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_52_02.jpg?sign=1739615482-iqlx0vMguuO1C1obanNZSxLjs3jjwyDQ-0-69d1552a5ae7d0d95dc9d19c112dbe85)
a.通过参数注入当前的CommandLineRunner Bean中。
b.CommandLineRunner是一个函数接口,输入的参数为main方法里接收的args参数。这里使用Lambda表达式执行每个Bean的doMyThing方法,如图3-1所示。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_53_01.jpg?sign=1739615482-PkcTqbOs99hnvyfUN5I3bZp2AtOaftRD-0-59ec16e281bc64033581e5e281e181e5)
图3-1
CommandLineRunner有个姊妹接口叫作ApplicationRunner,它们之间唯一的区别是ApplicationRunner使用org.springframework.boot.DefaultApplicationArguments类型的参数,示例如下。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_53_02.jpg?sign=1739615482-8WlupaY6fBPlnOUPqlfsAzfIDS8hDJZU-0-ed5dd969f45fcd93560bff8e931705af)
CommandLineRunner的args是不定长字符串(String... args),而ApplicationRunner的args是DefaultApplicationArguments类型的对象。
3.2.5 Bean的Scope
容器中的Bean的Scope指的是Bean的实例在容器中创建的方式。在容器中,默认是singleton,即整个容器中只创建一个Bean的实例。常用的还有prototype,即每次请求Bean时都会创建一个实例。可以通过@Scope注解来设置Scope。
下面两种方式是相同的:
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_53_03.jpg?sign=1739615482-0AUWbP5LOpGFQ91c8eGjiL4ab544Kpqz-0-a4c165436b6e55ba682cd96d5a55a359)
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_54_01.jpg?sign=1739615482-Sbt9e9tThkUOFwHkR3HnwiAMqojUgtQ5-0-6a3b6f1ecb55ee8c4e39873c11e63e1a)
通过@Scope(BeanDefinition.SCOPE_PROTOTYPE)指定Scope为prototype:
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_54_02.jpg?sign=1739615482-6rmnssTrLSxYmAJYKajkeSdM9wCSMWFO-0-7247b71588ef992ae02e0cb84f09a31e)
除可以在方法上注解@Scope外,还可以在@Bean的类上注解@Scope,示例如下:
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_54_03.jpg?sign=1739615482-JDkIV7Q6s8DDolDtN4SXgQ5hLX1JuHvu-0-e5006bee1bb0f54030e2fdf3cd74e0bb)
在JavaConfig中配置的代码如下:
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_54_04.jpg?sign=1739615482-fvvtw43cKNun2qE0lmuOCRlqYe9AKQaX-0-ce0ad0b860d05963fd636186677e0295)
这时可以在ScopeInjectService Bean中分别给上面三个Bean注入两次,由此判断相同类型的两个注入是否相等。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_54_05.jpg?sign=1739615482-nmNE6AcY96hfq4CpiNSXAmxwQaTzTrAO-0-d57ec00e134237c3ed69b90af65513fa)
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_55_01.jpg?sign=1739615482-O4KdAsgJLq0UPA00fjiT3O1Kbd9kWTgd-0-942d99027ea12a742a73b185ed3a6a1d)
在JavaConfig中配置下面的代码。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_55_02.jpg?sign=1739615482-ZkurSNKwtHVJ5eRxcz6smixah7WhonJz-0-f31ecd84bb5ba28b854859e1dbce387c)
执行效果如图3-2所示。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_55_03.jpg?sign=1739615482-kMR3k9lfs4OmtLzUBs1SfdtEw1jQiLAc-0-d7762e9611f1f6a6911b7a3ccf6b5402)
图3-2
3.2.6 Bean的生命周期
1.初始化和销毁
我们可以定制Bean在容器中的初始化行为和销毁行为。
(1)注解配置:使用@PostConstruct和@PreDestroy。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_55_04.jpg?sign=1739615482-l1oAjnf24YARKaDb7IkSZSqvmJvvrKRG-0-ce7fa116c8a1b4e3188afd388813565f)
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_56_01.jpg?sign=1739615482-8Qksgt2ZFBk6lN9OstLpjWLz2g8hEPmp-0-ced3368720b2b2726fe4ecc466bb200e)
执行效果如图3-3所示。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_56_02.jpg?sign=1739615482-7aZunBjJC85VFcj3Ytuw2ChvOBsbJXha-0-f4ca9b394bac5af9eea5a7a42c56d689)
图3-3
(2)Java配置:使用@Bean的initMethod和destroyMethod。
Bean的定义如下。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_56_03.jpg?sign=1739615482-EuHZWiNRuTiC62roz1Fctfe9xi0tJmQ0-0-f32eef26fb9ac8aa36bc4c52216e0f75)
在JavaConfig中配置下面的代码。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_56_04.jpg?sign=1739615482-jlqOcMuCjqcOy8a31YmMMGb4wBNxYkdu-0-c596dd715a55e3d9d08b418f01103c05)
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_57_01.jpg?sign=1739615482-J4KHupk0CVo2EOf6PTzsayD5PwL3l4N6-0-0c2a77d556d0c404166c5c14dd4f72ca)
执行效果如图3-4所示。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_57_02.jpg?sign=1739615482-v6KAHvhIHednoI7YYS6uRYW0IC71JmQP-0-7adc0fa90a68bb602682ffa1c08df4a3)
图3-4
2.延迟初始化(@Lazy)
只要在Bean上注解了@Lazy,那么Bean在被调用时就会被初始化。它可以和@Component类注解或@Bean一起使用。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_57_03.jpg?sign=1739615482-Kbf5NuJXovnQherPZ77KFKOL4UQKvQ7G-0-ebfe76e197ccd8eb1bfa22529f45ad8a)
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_57_04.jpg?sign=1739615482-BPH5ULOiVXbfUkNN4ZrRDFSxta5WaYCD-0-5568e59cbb80d5217c149a62026e5ad3)
因为这两个Bean没有被调用过,所以没有被初始化,此时控制台没有任何输出。
3.依赖顺序(@DependsOn)
设置Bean lifeService2依赖于lifeService,让lifeService先初始化,可以用@DependsOn来实现。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_57_05.jpg?sign=1739615482-g5my3YDJKceeyfmt4BKB1iDdNTB4Epkp-0-217091291ccf3edc5c32d6e00d9c91e3)
执行的结果是lifeService先于lifeService2初始化,如图3-5所示。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_58_01.jpg?sign=1739615482-QEClR3P0ocKBUKQplBvbDsFNfFhoT92o-0-8c478929ab5167d9124b3f01975a0cd7)
图3-5
3.2.7 应用环境
Spring提供了一个接口Environment来代表当前运行的应用环境,这个环境包含两部分。
◎Profile:一组命名的、定义在一起的Bean。通常为不同的应用场景(生产环境、开发环境、测试环境等)定义。
◎Property:配置属性,可以从properties文件、JVM系统属性、操作系统环境变量等外部来获得配置属性。
1.场景(@Profile)
可以通过@Profile注解指定当前的运行场景。@Profile可以和@Component、@Configuration、@Bean等一起使用,当然也分别限制了@Profile生效的Bean的分组。
下面使用需要显示不同操作系统的列表命令(在Windows下为 dir,在Linux下为 ls)的Bean。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_58_02.jpg?sign=1739615482-watjnPlUiNzSS6L78HroXyjGgmWTXeYx-0-8e9b2a306fd0a7e4bd15316dfd162dc6)
在Windows开发环境下,场景配置如下。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_58_03.jpg?sign=1739615482-m1NsgIlHcNBdC4ieOHo9knDlOdzbU96c-0-b38cf141db7747f1b0472608361052b9)
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_59_01.jpg?sign=1739615482-snh0pdcB9N9fg69o3WJkT84gDkzja9Zn-0-8d4cb0d742131c9ab0b5729f7a7bdd44)
在Linux开发环境下,场景配置如下。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_59_02.jpg?sign=1739615482-n9U7eaIsu2wPJ4pFofgcdHrcKP0SzdyJ-0-32659dc52a5d77272c8b6b408b0a6da7)
当配置好两种不同场景下的Profile后,我们需要在应用中配置哪个是激活的Profile,手动配置如下。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_59_03.jpg?sign=1739615482-OQ685H0hHWC84s0TyvZhs5KEAQ2UA6fC-0-d1aaa021576e4cd3025cf3648389a284)
因为使用了Spring Boot,所以只需在application.properties文件中做如下配置即可。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_59_04.jpg?sign=1739615482-xIQRmfg7ucHABfY93nQyv2976KAXA1Gg-0-39c9159953daf8953a0989abe76bc453)
在JavaConfig里,用CommandLineRunner分别将Profile配置成production和dev,执行效果如图3-6和图3-7所示。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_59_05.jpg?sign=1739615482-Gk3cPzAHWKeUkmFbRRwt2WRD2Ko1ULMe-0-9b24dbd24265ed3ecd963c61664b8d88)
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_59_06.jpg?sign=1739615482-jn6u4VdRnsLAF2ncv0jA0x4vdAMXkNQc-0-e3ddc2feb4a4dc9cb49b791b3b75ee54)
图3-6
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_59_07.jpg?sign=1739615482-T9il8MGi3NPhQe0gPLQ81BgHNmszKnTz-0-5d88cf81935870dd338c7bfc1d6ee795)
图3-7
2.属性配置(@PropertySource)
Spring的Environment属性是由PropertySource组成的,我们可以通过@PropertySource指定外部配置文件的路径。这些配置文件的属性都会以PropertySource的形式注册到Environment中,@PropertySource支持XML格式和properties格式,不支持Spring Boot下的YAML格式。
现在添加2个外部配置文件。
◎author.properties:
author.name=wyf
◎book.properties:
book.name=spring boot in battle
在添加完成后,可以用一个配置类来接收这两个文件的配置。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_60_01.jpg?sign=1739615482-3tuMGoxlGfddsoKxwueggCxF43tC5FlT-0-8109d69e5e2086fb0df002c150e0c9e5)
a.当有多个外部配置时,可以用@PropertySources指定。若只有一个可用,则只使用@PropertySource("classpath:book.properties")。
b.注入Environment的Bean,因为只有一个构造器,所以可省略@Autowired。
c.可以通过@Value注解获得Environment中的属性,关于@Value的更详细的讲解见3.6节。
d.外部配置的属性都已经在Environment中注册过,可以直接获取。
3.2.8 条件配置(@Conditional)
通过@Conditional我们可以定义当满足某个特定条件(Condition)时,应该做什么配置。@Conditional同样可以和@Component、@Configuration、@Bean一起使用,进而指定条件起作用的范围。
@Conditional注解接收Condition数组作为参数,Condition即我们的特定条件。Condition只有一个方法matches,当符合条件时,返回true;当不符合条件时,返回false。
例如,判断当前系统是否是Windows的条件定义:
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_61_01.jpg?sign=1739615482-wKyv9TwP2J6PetiNEko6q5mjKCHDAZvB-0-e4d5a4f1d798e779422c35e1b2fbb814)
a.条件实现Condition接口即可。
b.matches的两个参数:ConditionContext可获得容器的相关信息;AnnotatedTypeMetadata是当前被注解的方法或类的元数据(数据的描述)信息。
c.通过容器context获得运行环境Environment信息,从而获得操作系统信息。
配置如下。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_61_02.jpg?sign=1739615482-xRMfKfZy6p36Dc8xe3TSyxxW0CGBEbf1-0-aaffa8834e71ece6fd904904fc44af20)
a.@Conditional使用的是OnWindowsCondition条件,只有在操作系统是Windows的情况下,当前Bean才会被创建。
在JavaConfig中使用CommandLineRunner运行。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_61_03.jpg?sign=1739615482-qyGoQUoSJCJeojGm8MuOoYuIJmdtDwff-0-969b60fa28621642dbbf3ef445e2eae8)
在Windows系统才能正常执行;在非Windows系统下会报错,找不到Bean。因为不符合条件,所以没有创下这个Bean。
3.2.9 开启配置(@Enable*和@Import)
在本书后面的内容里会出现大量以@Enable*开头的注解,@Enable*会自动对相应的功能进行自动配置,如@EnableWebMvc、@EnableCaching、@EnableScheduling、@EnableAsync、@EnableWebSocket、@EnableJpaRepositories、@EnableTransactionManagement、@EnableJpaAuditing和@EnableAspectJAutoProxy等。
@Enable*的开启配置的功能依赖于@Import注解,@Import注解支持导入如下配置:
◎直接导入@Configuration配置类。
◎配置类选择器ImportSelector的实现。
◎动态注册器ImportBeanDefinitionRegistrar的实现。
◎混合以上三种。
下面将分别演示四种方式的实现。
1.直接导入@Configuration配置类
当应用注解了@Configuration后,会被Spring Boot的默认组件扫描并自动注册,所以本节的注解类代码放在 io.github.wiselyman.annotations中,配置类的代码放在io.github.wiselyman.config中。
定义注解:
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_62_01.jpg?sign=1739615482-lfSTd7nbRevRvnvFPp1O4Zoudg1fmfq0-0-41d368ae421cf2da840570be2d52cb3e)
定义配置类:
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_62_02.jpg?sign=1739615482-5ssBg78qi4bE1DQdF4tEGYAUyeBw9X0t-0-d44ee74acb41f9f999b389c3bbe18458)
在JavaConfig中使用@EnableA注解,即可获得导入的配置类AConfig中的Bean a。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_63_01.jpg?sign=1739615482-gPtQS1OXEGAkPcL7W1Xom28vu26t20FX-0-49a4311dd053ff3a1cee6c13cfa6fb05)
在JavaConfig中使用CommandLineRunner查看Bean的内容,执行结果如图3-8所示。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_63_02.jpg?sign=1739615482-neP29ljvTKcxKpMG31ZpQEOWhqjdyu2L-0-283b95dd6bc38bab54f973503ed28ffc)
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_63_03.jpg?sign=1739615482-VlhDYPtwNhM4iwQZnmaonJCEAD24Gw90-0-2565686779fa5e4f30c67d8ce10ed997)
图3-8
2.配置类选择器ImportSelector的实现
在这个例子中,通过注解选择生效的配置类,注解定义如下。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_63_04.jpg?sign=1739615482-becndIs3wFDI8cXOERAHkKSeMdv6ZKvf-0-974e299e6c5428fbfdeeb54474116400)
在io.github.wiselyman.selector中定义选择器。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_63_05.jpg?sign=1739615482-2q2uP2QBHtFJ1BToXgaO1xCh5mvFTXDY-0-e360294e918d8058a828c7a000fd58e1)
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_64_01.jpg?sign=1739615482-L3XMZVAthcW2aPAWmypy2HXuEvvMJaUc-0-5053f28bd111f0409fe28cb4f89b8c34)
a.选择器要实现ImportSelector接口。
b.实现接口的selectImports方法,参数AnnotationMetadata importClassMetadata是注解使用类(本例为JavaConfig)上@EnableB的元数据信息。
c.通过@EnableB在实际使用中的元数据,获得isUppercase的值。
d.如果isUppercase==true,则此时实际使用的是@EnableB或者@EnableB(isUppercase=true),因而使用BUppercaseConfig提供的配置。
e.若实际使用的是@EnableB(isUppercase=false),则使用BLowercaseConfig提供的配置。
BUppercaseConfig的定义如下。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_64_02.jpg?sign=1739615482-GBszqItBeu3fYHk3JgLJffWYlEm3WjCa-0-51ee17c408621c0219398b82102ce1cc)
BLowercaseConfig的定义如下。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_64_03.jpg?sign=1739615482-gdAVttrxLKr1yxM3yd4ViTMzYXOZv1AR-0-506fae2f0212e971192f5d40fccb4cee)
在JavaConfig中使用@EnableB,并用CommandLineRunner检验。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_64_04.jpg?sign=1739615482-vo2VC5Zwtj1lWWZbWhtMKWdLe9ydo6Ak-0-6437786819b3be71b718288f570e7528)
运行结果如图3-9所示。
若将isUppercase设置为false,则执行结果如图3-10所示。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_64_05.jpg?sign=1739615482-n5muXleUk7n61vTJod4SxS4jKoTtepe6-0-d7891a3133c2673f25426c691ca77831)
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_65_01.jpg?sign=1739615482-sBACy4c1FockUsTfvJ7OXOeNufQfR8hf-0-a969d57beddfad632465efda5cca4801)
图3-9
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_65_02.jpg?sign=1739615482-Mt2cVmNqOUjGE9c7MpmuPJ06jRazy1qo-0-c8757978180a5317a47aa17004256c10)
图3-10
3.动态注册器ImportBeanDefinitionRegistrar的实现
本例通过ImportBeanDefinitionRegistrar动态注册Bean到容器里。
注解定义如下:
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_65_03.jpg?sign=1739615482-7OitTMRO5DcDx4V3BidBb3g8JM2fFgNS-0-b0810a2ba7b46c4fec0154bd6d4717ab)
在io.github.wiselyman.registrar中定义注册器。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_65_04.jpg?sign=1739615482-TyS9lSDSqErLK0lD7wEiyWXNSONzoJar-0-599c0bea97b4c35e9396b82dab4b6286)
a.注册器需实现ImportBeanDefinitionRegistrar接口。
b.实现registerBeanDefinitions参数AnnotationMetadata importClassMetadata是注解使用类(本例为JavaConfig)上@EnableB的元数据信息。
c.参数BeanDefinitionRegistry registry用来注册所有Bean的定义的接口。
d.可以使用BeanDefinitionBuilder来编程实现Bean的定义(BeanDefinition),此句定义了一个类型为String的Bean。
e.构造String的值是C。
f.设置Bean的Scope是singleton。
g.获得Bean的定义。
h.将Bean注册为名称为c的Bean。
此时,在JavaConfig上使用@EnableC注解,并用CommandLineRunner进行检验。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_66_01.jpg?sign=1739615482-dxsBsi3gjm1LvzvaDqrhJN6ymluG2qut-0-2d4a600dc6294149aa2076a3a2d931ab)
IntelliJ IDEA可以检测到静态注册的Bean,但检测不到动态注册的Bean,因而IDE会标识红色,如图3-11中方框所示。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_66_02.jpg?sign=1739615482-sXrhXogDQ498CAkVF6YuSVQSdRbveUdl-0-0ed2ac8d7f8e5fc50533a92566e4a06f)
图3-11
但可以正常运行,运行结果如图3-12所示。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_66_03.jpg?sign=1739615482-DGJ2YHNIrG7esFULMaQi21EEdDjSooAe-0-d48b1c79928ddd331a2b47dd4b1328ba)
图3-12
4.混合使用
@Import支持导入配置类的数组,因而我们可以混合上面三种配置,定义一个注解,使其具备上面三个功能。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_66_04.jpg?sign=1739615482-jE4w7cIvjLjSmUgsXXCJP0kK75McrROF-0-5da66ec4e26f8c95d5e8e5e19765dc97)
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_67_01.jpg?sign=1739615482-2vvtuZU7o1tGviwczQo1jRbjPcCV4gvY-0-b532bb9c40cb7cb352065d3403cd0aca)
因为选择器里指定了要使用的注解的类,所以需要新建一个选择器。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_67_02.jpg?sign=1739615482-AnpA8fgawblbJOPQRUWRxoIDPyld4DQr-0-5e25fc164117feb3fadb8645e8e0fd74)
在JavaConfig中启用@EnableABC,并用CommandLineRunner进行检验。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_67_03.jpg?sign=1739615482-jFQjnZlarYAano04CLeySMm7Tj766qOm-0-e98737922b9f5f9fa64dd88e5e111ab8)
校验结果如图3-13所示。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_67_04.jpg?sign=1739615482-l7JPV65UQowaHdMuiuTRauY0it0J18Bm-0-5d4faeb16da19c90b02d3a5073ce725f)
图3-13