使用 Sping Boot 创建一个自定义的 Auto-Configuration
正文
Create a Custom Auto-Configuration with Spring Boot
1. Overview
Simply put, the Spring Boot autoconfiguration represents a way to automatically configure a Spring application based on the dependencies that are present on the classpath.
自动化配置简言之,就是一个基于在 classpath 里面的依赖,自动化配置一个 Spring 程序的道路。
This can make development faster and easier by eliminating the need for defining certain beans that are included in the auto-configuration classes.
In the following section, we’re going to take a look at creating our custom Spring Boot auto-configuration.
2. Maven Dependencies
Let’s start with the dependencies that we need:
|
1 2 3 4 5 6 7 8 9 10 |
|
The latest versions of spring-boot-starter-data-jpa and mysql-connector-java can be downloaded from Maven Central.
3. Creating a Custom Auto-Configuration
To create a custom auto-configuration, we need to create a class annotated as @Configuration and register it.
先创建一个类,使用 @Configuration 注解来注册它。
Let’s create a custom configuration for a MySQL data source:
|
1 2 3 4 |
|
The next mandatory step is registering the class as an auto-configuration candidate, by adding the name of the class under the key org.springframework.boot.autoconfigure.EnableAutoConfiguration in the standard file resources/META-INF/spring.factories:
下一步,把这个类注册成一个 auto-configuration 候选人,在 resources/META-INF/spring.factories 这个标准文件里面,增加在 org.springframework.boot.autoconfigure.EnableAutoConfiguration 这个后面。
|
1 2 |
|
If we want our auto-configuration class to have priority over other auto-configuration candidates, we can add the @AutoConfigureOrder(Ordered.HIGHEST_PRECEDENCE) annotation.
这里可以有多个候选者,如果想提升我们的这个候选者的优先级,我们需要增加 @AutoConfigureOrder(Ordered.HIGHEST_PRECEDENCE) 这个注解。
Auto-configuration is designed using classes and beans marked with @Conditional annotations so that the auto-configuration or specific parts of it can be replaced.
Auto-configuration 是这样设计的,可以使用 @Conditional 注解的 classes 和 beans, 这样它的 auto-configuration 或者其他部分可以被替代。(这里怎么理解?)
Note that the auto-configuration is only in effect if the auto-configured beans are not defined in the application. If you define your bean, then the default one will be overridden.
注意,auto-configuration 仅仅在 auto-configured 的beans 没有被应用程序定义的情况下生效。如果你定义了你的 bean,那么这个默认的将会被覆写。这里怎么理解?)
3.1. Class Conditions
Class conditions allow us to specify that a configuration bean will be included if a specified class is present using the @ConditionalOnClass annotation, or if a class is absent using the @ConditionalOnMissingClass annotation.
Class conditions 允许我们使用 @ConditionalOnClass 去指定,如果一个指定的类存在,则一个配置的bean将会包含,反之,则使用 @ConditionalOnMissingClass。
Let’s specify that our MySQLConfiguration will only be loaded if the class DataSource is present, in which case we can assume the application will use a database:
|
1 2 3 4 5 |
|
3.2. Bean Conditions
If we want to include a bean only if a specified bean is present or not, we can use the @ConditionalOnBean and @ConditionalOnMissingBean annotations.
To exemplify this, let’s add an entityManagerFactory bean to our configuration class, and specify we only want this bean to be created if a bean called dataSource is present and if a bean called entityManagerFactory is not already defined:
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 |
|
Let’s also configure a transactionManager bean that will only be loaded if a bean of type JpaTransactionManager is not already defined:
|
1 2 3 4 5 6 7 |
|
3.3. Property Conditions
The @ConditionalOnProperty annotation is used to specify if a configuration will be loaded based on the presence and value of a Spring Environment property.
@ConditionalOnProperty 这个注解被用于指定,如果一个配置将会被加载,仅当一个 Spring Environment property 存在或者值存在。
First, let’s add a property source file for our configuration that will determine where the properties will be read from:
|
1 2 3 4 |
|
We can configure the main DataSource bean that will be used to create connections to the database in such a way that it will only be loaded if a property called usemysql is present.
We can use the attribute havingValue to specify certain values of the usemysql property that have to be matched.
Let’s define the dataSource bean with default values that connect to a local database called myDb if the usemysql property is set to local:
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 |
|
If the usemysql property is set to custom, the dataSource bean will be configured using custom properties values for the database URL, user, and password:
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 |
|
The mysql.properties file will contain the usemysql property:
|
1 |
|
If an application that uses the MySQLAutoconfiguration wishes to override the default properties, all it needs to do is add different values for the mysql.url, mysql.user and mysql.pass properties and the usemysql=custom line in the mysql.properties file.
3.4. Resource Conditions
Adding the @ConditionalOnResource annotation means that the configuration will only be loaded when a specified resource is present.
@ConditionalOnResource 意味着配置文件仅仅被加载,当指定的资源存在时。
Let’s define a method called additionalProperties() that will return a Properties object containing Hibernate-specific properties to be used by the entityManagerFactory bean, only if the resource file mysql.propertiesis present:
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 |
|
We can add the Hibernate specific properties to the mysql.properties file:
|
1 2 3 |
|
3.5. Custom Conditions
If we don’t want to use any of the conditions available in Spring Boot, we can also define custom conditions by extending the SpringBootCondition class and overriding the getMatchOutcome() method.
我们可以通过扩展 SpringBootCondition 类来自定义条件,覆写 getMatchOutcome() 方法,从而达到我们自定义的条件判断。
Let’s create a condition called HibernateCondition for our additionalProperties() method that will verify whether a HibernateEntityManager class is present on the classpath:
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 |
|
Then we can add the condition to the additionalProperties() method:
|
1 2 3 4 |
|
3.6. Application Conditions
We can also specify that the configuration can be loaded only inside/outside a web context, by adding the @ConditionalOnWebApplication or @ConditionalOnNotWebApplication annotation.
4. Testing the Auto-Configuration
Let’s create a very simple example to test our auto-configuration. We will create an entity class called MyUser, and a MyUserRepository interface using Spring Data:
|
1 2 3 4 5 6 7 |
|
|
1 2 |
|
To enable auto-configuration, we can use one of the @SpringBootApplication or @EnableAutoConfiguration annotations:
|
1 2 3 4 5 6 |
|
Next, let’s write a JUnit test that saves a MyUser entity:
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 |
|
Since we have not defined our DataSource configuration, the application will use the auto-configuration we have created to connect to a MySQL database called myDb.
The connection string contains the createDatabaseIfNotExist=true property, so the database does not need to exist. However, the user mysqluser or the one specified through the mysql.user property if it is present, needs to be created.
We can check the application log to see that the MySQL data source is being used:
|
1 |
|
5. Disabling Auto-Configuration Classes
If we wanted to exclude the auto-configuration from being loaded, we could add the @EnableAutoConfiguration annotation with exclude or excludeName attribute to a configuration class:
|
1 2 3 4 5 6 |
|
Another option to disable specific auto-configurations is by setting the spring.autoconfigure.excludeproperty:
|
1 |
|
6. Conclusions
In this tutorial, we’ve shown how to create a custom Spring Boot auto-configuration. The full source code of the example can be found over on GitHub.
The JUnit test can be run using the autoconfiguration profile: mvn clean install -Pautoconfiguration.
参考
https://www.baeldung.com/spring-boot-custom-auto-configuration
posted on 2019-12-15 17:05 chaiyu2002 阅读(237) 评论(0) 收藏 举报
浙公网安备 33010602011771号