![从企业级开发到云原生微服务: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=1739277406-aL9QLjAExprPoyhZbX0JNKdZQ19ZvaRj-0-80b91380bd8c80ffacd225d8d7683923)
@Component、@Service、@Repository和@Controller在当前示例中是完全等同的。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_46_03.jpg?sign=1739277406-0xnkcxOxnxb7961KvIeA7DILpRzZRdZU-0-9ca4e5bfa807fed2afaf69f9cdd7edc0)
上面的@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=1739277406-HoNsNCH1NBblKAPKOeqN4rviFntGJh3i-0-16e741a98266e3c178e248578d2bfb26)
用Java配置的如下。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_47_02.jpg?sign=1739277406-Dm9pgK5LboilZ2wGkV7nawhEVxbW650a-0-f57740a6fb09b6e7e35806139c977950)
同样,没有给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=1739277406-jtYMJLmXUcGuZH5aWfFUXANpdytzdbQZ-0-78ea2fb2f98fb738dc522da207cb1a6e)
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_48_01.jpg?sign=1739277406-zDy4CKzUEI6UmaP4djaN9G9YpVVtxewI-0-8f52e0e2fcdd2e248b5b638e602edfb3)
在构造器上注解注入是Spring推荐的注入方式,当然,也可以通过在属性上注解@Autowired来注入Bean。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_48_02.jpg?sign=1739277406-4XUmliWCmSst7i0Y4AAqt3G3nvjJCtbZ-0-d6897712dba3277ca0b80d6d4950bb64)
还可以在set方法上注解@Autowired来注入Bean。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_48_03.jpg?sign=1739277406-q9UqkhzLYM35wSA1kA4X3R3Fw4goLPJ3-0-0b7bcc6c00e60e08f94d5f512059a6e1)
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_49_01.jpg?sign=1739277406-x6z274JCMIOWJnBftBpXGwyFFzD45Ehp-0-8ad477ed7d982dd9664ce5bdd9bcc4d0)
如果Bean只有一个构造器,则可以直接省略@Autowired注解。若Bean有多个构造器,则需注解一个构造器用来注入,示例如下。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_49_02.jpg?sign=1739277406-nB6sssDQjA1Myie4hqN8ZjgcQB6Pmbwn-0-145c0b9e6cd7c48f30d32c68a0244c66)
(2)配置注入。
现在使用Java配置的方式在Bean JavaConfigInjectService中注入BeanAnotherService,JavaConfigInjectService定义如下。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_49_03.jpg?sign=1739277406-B0hb6nAGpTRwUrSY7jYMEOwRGIrY4egh-0-664d085ebe55d139be637bc4507f1892)
前面已经将AnotherService通过@Bean注解成Bean了,下面只需在定义JavaConfigInjectService的Bean的方法参数里注入AnotherService的Bean即可。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_49_04.jpg?sign=1739277406-GyU04OaG4tUmNFBTjeWUJu08qkgZ0s6v-0-37f518a408ef1dbcd02bde5be407f9b4)
在同一个配置类里,还可以在新建JavaConfigInjectService的构造里直接注入创建SomeService2的Bean的方法。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_49_05.jpg?sign=1739277406-cauDIX1qKHeeOubtxxeTeKDTOQxi3AUy-0-3cd04fd86ae5ac727c99b4f4a02bd42f)
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_50_01.jpg?sign=1739277406-iJNJs0OU94YdiTB1OVFgiID8hYqpOkov-0-89b20f1057a6f2fe878e38168ec5f4e8)
(3)混合注入。
注解配置的Bean可以直接注入给使用Java配置的Bean,反之亦然。
把注解配置的Bean注入Java配置的Bean:
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_50_02.jpg?sign=1739277406-IIaB71QTnVFkD6LzjFZpoX2WkaEuAQ25-0-5dd6542cdeaebd49d2b340fef13e1aba)
把Java配置的Bean注入注解配置的Bean。被注入的BeanMixInjectionService2定义如下。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_50_03.jpg?sign=1739277406-38M09E8B26yHF84nTX2cFnHPa9iOyRPB-0-0c8eb428924607f7f5a0af18912e16ef)
在JavaConfig类里,可以直接在参数中注入Bean。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_50_04.jpg?sign=1739277406-8YPxnt3WmgHpCHGkEBRHlgJzXhmQgHEt-0-0847243a50ed42e8288736c09b9df623)
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=1739277406-wHL7jYaPRvemit3sA6oFekwNGzm68noY-0-861d8eec7fa44444fd1de1e7e1a5d7e9)
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_51_01.jpg?sign=1739277406-gp6oqYy52kk2fjLj8ptgHsF0yqZm9DuZ-0-f4228fb539331c6c30016e1ed9db7b17)
此时有两个Bean,名称分别为anotherService和primaryAnotherService。如果在注入的地方不使用这个两个名称,那么就会按照Bean的类型自动注入。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_51_02.jpg?sign=1739277406-IqH7KTxBw42eQrmDY25lKaLGhiMUS3h6-0-8884bd8583aacb39b4a2a5df5f77c98c)
因为现在使用的 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=1739277406-Wy0mASM00pIb1zbq2VbZHW6YN5eFBG8V-0-d2f21b4b8809ba63ccd7ae00a28c3caf)
注入primaryAnotherService。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_52_01.jpg?sign=1739277406-CJS6G24QKk13oZXQyEWg8V9feQVhmIAx-0-8bce54f63aa13a6353cc73048b89c329)
3.2.4 运行检验(CommandLineRunner)
在Spring Boot下可以注册一个CommandLineRunner的Bean,在容器启动后,这个Bean可用来执行一些专门的任务,如在JavaConfig里。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_52_02.jpg?sign=1739277406-S8ZMAeis5bp4QSSGtecYFI58u8VUOULE-0-f7c400d5b20477601e15e5f86a9e8688)
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=1739277406-2Y3jRFAW7Pheef3v9GtnaTje9cz9vBjp-0-97ccf846592ebf3aa57f4f88929c50aa)
图3-1
CommandLineRunner有个姊妹接口叫作ApplicationRunner,它们之间唯一的区别是ApplicationRunner使用org.springframework.boot.DefaultApplicationArguments类型的参数,示例如下。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_53_02.jpg?sign=1739277406-wtzSdePoh28nOZLBl4pV6qAGGiPruWuY-0-e586c8551e3e18eef2b26927e638abe6)
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=1739277406-mh5PBYbJKGkfz8vUT7j9zbgOt3HiymuF-0-a4ae97a6b8826ac03645b14484f01d3e)
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_54_01.jpg?sign=1739277406-0F1pqTUy4HQP6uOYFheyTfaP6XjVfHpu-0-fb3de8f28d9ae1fc9f2a0ec93b18904f)
通过@Scope(BeanDefinition.SCOPE_PROTOTYPE)指定Scope为prototype:
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_54_02.jpg?sign=1739277406-ShI7AOy8THBCwfEyNBHHnj9BUKenwkx0-0-e517aca8604bd532f09a0d20014abaf0)
除可以在方法上注解@Scope外,还可以在@Bean的类上注解@Scope,示例如下:
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_54_03.jpg?sign=1739277406-igWKryDMnqoSlTPVAKARtKKdwQohPNSO-0-b992de3df28896a81bf684dccdfb9972)
在JavaConfig中配置的代码如下:
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_54_04.jpg?sign=1739277406-NNmd2BuZZ2vfV7qQgdoeg7KjIM9qKzI0-0-ffa9bcdb9be3b229dfb002a0eb195889)
这时可以在ScopeInjectService Bean中分别给上面三个Bean注入两次,由此判断相同类型的两个注入是否相等。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_54_05.jpg?sign=1739277406-S6nfQhm08coPmq4qDZ7pIYKHLFqR9hNB-0-06df94912e281cb5436287f18da6eeae)
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_55_01.jpg?sign=1739277406-zZZo1Rgbnzx85XupfVrSHfMoqHfQcvTM-0-86297a466178f9b3de504b35cc505888)
在JavaConfig中配置下面的代码。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_55_02.jpg?sign=1739277406-78zpLSFUDRCKqGqONSvb0lBspp8GRiOb-0-15fe9c9d41fe5e2acccdd785140977fa)
执行效果如图3-2所示。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_55_03.jpg?sign=1739277406-m5Hfud4AdgsQiXmSyYvj3tW3xpNUqsYz-0-1fafcf3e5a2fe510c331a64fa76eeaf4)
图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=1739277406-ihFEAg4XOntNwuRiFPuEdkhEtDM9Fbtd-0-ba2909102fb04bc6835eba6e3cd7f0e2)
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_56_01.jpg?sign=1739277406-0IcPLXW116HaIOAM0vVXgyIcSKKPchIg-0-5d15f4b866b91d5e0375507d05a89196)
执行效果如图3-3所示。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_56_02.jpg?sign=1739277406-sXFPAYGADuW6ELIy2l1vrWwgdt0C4u9w-0-4b5e032da6d280389ff0dce00fc64a39)
图3-3
(2)Java配置:使用@Bean的initMethod和destroyMethod。
Bean的定义如下。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_56_03.jpg?sign=1739277406-oSl9tjCKHY2SWFMgW77Pe9fovtE8ghh4-0-dbf4454f55989fb60bc4399e4b563170)
在JavaConfig中配置下面的代码。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_56_04.jpg?sign=1739277406-wLCkvz0avDMalF7yZ7IMAlIClcv8rjUR-0-f43dff8c80be8f11960e46e118ac2613)
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_57_01.jpg?sign=1739277406-MRALv9joVqN9rVKNikXmiVACnkqNuqRm-0-c138e5c56a842769d86abc65bb64e3a6)
执行效果如图3-4所示。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_57_02.jpg?sign=1739277406-DnhMXz1N5Dp52RLTPfnG2gQSf3GJrZgF-0-bf400a1bf5cc63fabe70c1208d9b3cdb)
图3-4
2.延迟初始化(@Lazy)
只要在Bean上注解了@Lazy,那么Bean在被调用时就会被初始化。它可以和@Component类注解或@Bean一起使用。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_57_03.jpg?sign=1739277406-YW3l3kzg1xNhAyLPBQtqseK9cG2odYcD-0-794f550033ed3edaaddeaf98f3aaedf7)
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_57_04.jpg?sign=1739277406-MjmXAX3DW8yQIyDDLrMTTI0c46mYLL7p-0-f2e14eb56ced2b72b5ec4633de22583a)
因为这两个Bean没有被调用过,所以没有被初始化,此时控制台没有任何输出。
3.依赖顺序(@DependsOn)
设置Bean lifeService2依赖于lifeService,让lifeService先初始化,可以用@DependsOn来实现。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_57_05.jpg?sign=1739277406-LFYaurE1LAslEmh8CDyz2xuNbMbnlcLV-0-e252cf81499f40e2c87e7ebe35ad967c)
执行的结果是lifeService先于lifeService2初始化,如图3-5所示。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_58_01.jpg?sign=1739277406-oh8u5Fx4EJwTxc1UaRDWf7oR4y3dZW9B-0-443ee344423ce383a3d1fa3b88f51519)
图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=1739277406-yUEdI2h74Ki7yEm8R6GjGksN80zkP4Ms-0-c2bb9792c1f429f076abc2d2aa5b59cf)
在Windows开发环境下,场景配置如下。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_58_03.jpg?sign=1739277406-OBIu5cs1qQMvVkNcJmzo4484mx4A7sAd-0-f36606891260e01efc861da367e627b8)
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_59_01.jpg?sign=1739277406-0TqQ4ZgpAMCJPyceAH0ubHbt4KekSJer-0-465ec1f1b83cd08f52b7bb7f5ae5b88f)
在Linux开发环境下,场景配置如下。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_59_02.jpg?sign=1739277406-KX0j8PqYyvxPxucNexKcslydkmOAMnFf-0-25bdc1bf436e939a4033065dcaf875dd)
当配置好两种不同场景下的Profile后,我们需要在应用中配置哪个是激活的Profile,手动配置如下。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_59_03.jpg?sign=1739277406-qz49hukoxqpcCGa2Sr2pz6X9kUyqBVVD-0-9005c5d20dad24fd08302e8823e90cb3)
因为使用了Spring Boot,所以只需在application.properties文件中做如下配置即可。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_59_04.jpg?sign=1739277406-F2kaB8Rm4q6OGhuudysXZgSlFQ0HiUXh-0-390f2999eb5e5ead30154dc0fa0cd810)
在JavaConfig里,用CommandLineRunner分别将Profile配置成production和dev,执行效果如图3-6和图3-7所示。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_59_05.jpg?sign=1739277406-QmsLYxpHCmuqmggmvi9uyzJ6OJc2aIqd-0-57cfd81b4fb524f20acd7d6823a18cf3)
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_59_06.jpg?sign=1739277406-UsnbwQuCTFbW6BQxa66KsipoPKtbJUR5-0-55f33eb58523d40bce82f828d806e66c)
图3-6
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_59_07.jpg?sign=1739277406-G5idawjujMdAUe7ORhaFWtlG13d7W1Cq-0-2b919c0f9151b8e6d62a3dc9d96e3f2a)
图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=1739277406-Q9xIGN177oOMCvnoohl7P7HLj0Q0YLuo-0-d85e3e1792a3bb05e5f32216da46d7f1)
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=1739277406-fEF6CyxewRq1lUfuQhGcDo5yzv1KwDBh-0-0e819d8484c7dd0c83c4966520855add)
a.条件实现Condition接口即可。
b.matches的两个参数:ConditionContext可获得容器的相关信息;AnnotatedTypeMetadata是当前被注解的方法或类的元数据(数据的描述)信息。
c.通过容器context获得运行环境Environment信息,从而获得操作系统信息。
配置如下。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_61_02.jpg?sign=1739277406-yXtsAwZWOlAjan5qGBS8VR2NVJs6L1dO-0-5629504b1d9d2adb955efe1e1fce0282)
a.@Conditional使用的是OnWindowsCondition条件,只有在操作系统是Windows的情况下,当前Bean才会被创建。
在JavaConfig中使用CommandLineRunner运行。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_61_03.jpg?sign=1739277406-zr8G1aar1SgYOPVd7we7y7KsmcYRBZBA-0-130255ffda2f966341e85d838137700d)
在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=1739277406-7vrUhSmQ64TLCcM9Yb1bohXK0P89EpXm-0-09bc1e4fed00130069fc6c58110dd8ea)
定义配置类:
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_62_02.jpg?sign=1739277406-TA6UlxLq7QJ8Mjk8FwbIHWHGRE9Y7Pl8-0-bfd0f850af0aaff667fd38e8d41fa536)
在JavaConfig中使用@EnableA注解,即可获得导入的配置类AConfig中的Bean a。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_63_01.jpg?sign=1739277406-l9fSt6TqAuGgd8GXhzqRBqE0Peavzj8V-0-338600e46af609b8f3221db2d7a23b0d)
在JavaConfig中使用CommandLineRunner查看Bean的内容,执行结果如图3-8所示。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_63_02.jpg?sign=1739277406-kgPb7JIxT9bLAya0V4sZQez0Tx7WgE2Y-0-bdb92e837b3ff2bcfd101028ba7c7fe6)
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_63_03.jpg?sign=1739277406-ONiy3LWkjhyxhBytnZ0VCXIdoM89nESu-0-b1717bd8f0aff0414f24c0bd5096aa52)
图3-8
2.配置类选择器ImportSelector的实现
在这个例子中,通过注解选择生效的配置类,注解定义如下。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_63_04.jpg?sign=1739277406-zq4J31joXcNAeqO6vyB48MRiuXy9RT1Z-0-7b5c59075314c131136a991181db2ce5)
在io.github.wiselyman.selector中定义选择器。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_63_05.jpg?sign=1739277406-ixe7SAyXvH1H4Yd5Qs3OZvkuxMAmsbZN-0-0da235df8493511a587c57babb9a4d57)
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_64_01.jpg?sign=1739277406-FLAEBC1OSnNdN8GsyAmNJFUbZIXlFiyz-0-22b4b99ab7c24dc437e15df6b4c65ca0)
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=1739277406-gffz7aGowK9TNgskwtEb0j3AJ87ea7du-0-02c162ecc6869d1d850b900224183eb1)
BLowercaseConfig的定义如下。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_64_03.jpg?sign=1739277406-2Ck4AOLQMemaooMsm9DzKIofrQuMAdAT-0-e5ba232a773190667144c51f971ae4fd)
在JavaConfig中使用@EnableB,并用CommandLineRunner检验。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_64_04.jpg?sign=1739277406-GP3tM0TA1yJI6CJPJ7p4u00uBRFeuh8A-0-5f65705107620102c9e8b4ebd4f5c351)
运行结果如图3-9所示。
若将isUppercase设置为false,则执行结果如图3-10所示。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_64_05.jpg?sign=1739277406-UYPmspHg1wxZlxgbHeaQpbBpxPtRPJNC-0-bb8c3b99b0c21cb72ddc3f921c11e3cb)
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_65_01.jpg?sign=1739277406-J1WChPqqKWv7uq4qIlLkEmtSgClXLXwY-0-22a87890a30255400e3f96863ff5f216)
图3-9
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_65_02.jpg?sign=1739277406-eCt4puHwqvZmv92jXjYHH5lCzKy4hBue-0-ee895ae696904c8babf775ad3240fa77)
图3-10
3.动态注册器ImportBeanDefinitionRegistrar的实现
本例通过ImportBeanDefinitionRegistrar动态注册Bean到容器里。
注解定义如下:
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_65_03.jpg?sign=1739277406-vsn67Wkov6Jh2sdope8wHKl8XIBZgfrY-0-5ed5a28bf267738822ce340a3d26845f)
在io.github.wiselyman.registrar中定义注册器。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_65_04.jpg?sign=1739277406-rMapoimkvbrWhH09i1bGMhXyeHqO1swA-0-32401db04868cf481b63ba7214b28fd8)
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=1739277406-mT3nQl9abWnxk5w25bMhFDlbvlzbyflk-0-1164d388b8296df412b700be9683a76c)
IntelliJ IDEA可以检测到静态注册的Bean,但检测不到动态注册的Bean,因而IDE会标识红色,如图3-11中方框所示。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_66_02.jpg?sign=1739277406-sxAbc1AwS8v5ExT8DcKksjrxgA6ukUv9-0-1ffa6f8d8e89c66273d56a75cddbd749)
图3-11
但可以正常运行,运行结果如图3-12所示。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_66_03.jpg?sign=1739277406-DxkeQ5tLXAe9VklaRZulxQOO8L47wDiN-0-c1d3f1759c0a1f9e8abd14205608d7f9)
图3-12
4.混合使用
@Import支持导入配置类的数组,因而我们可以混合上面三种配置,定义一个注解,使其具备上面三个功能。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_66_04.jpg?sign=1739277406-OT1hImHmQ8cYIVwihklbkV3NMRLvN1Mo-0-4be9bbc6ea6821a05db8548daa7284c2)
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_67_01.jpg?sign=1739277406-uTqClIkfvIGyAd96KIM4nJOhISHBt2Hz-0-73089c98605284b30327683a2d0c6e78)
因为选择器里指定了要使用的注解的类,所以需要新建一个选择器。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_67_02.jpg?sign=1739277406-Fehrcu3uqrek0WifPrUwSELjLNLXHjHy-0-85971ef1ddd6ceacd8d77479c81449ef)
在JavaConfig中启用@EnableABC,并用CommandLineRunner进行检验。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_67_03.jpg?sign=1739277406-ejWOY2CG15h9BrPL9Fpoj8oCNBfwUjTC-0-06e57a7192127ff91593b047e3bc8d9f)
校验结果如图3-13所示。
![](https://epubservercos.yuewen.com/E5E2EB/18096059808236406/epubprivate/OEBPS/Images/37792_67_04.jpg?sign=1739277406-89tnOCrZbVPkW7E1y0tyUqZY1IaVmIz0-0-cd187223ca545afa533145cdad7a7ffb)
图3-13