怎樣通過SpringBoot整合Mybatis分析自動配置

這期內(nèi)容當中小編將會給大家?guī)碛嘘P(guān)怎樣通過SpringBoot整合Mybatis分析自動配置,文章內(nèi)容豐富且以專業(yè)的角度為大家分析和敘述,閱讀完這篇文章希望大家可以有所收獲。

成都創(chuàng)新互聯(lián)公司成立于2013年,先為牟定等服務(wù)建站,牟定等地企業(yè),進行企業(yè)商務(wù)咨詢服務(wù)。為牟定企業(yè)網(wǎng)站制作PC+手機+微官網(wǎng)三網(wǎng)同步一站式服務(wù)解決您的所有建站問題。

前言

SpringBoot憑借"約定大于配置"的理念,已經(jīng)成為最流行的web開發(fā)框架,所以有必須對其進行深入的了解;本文通過整合Mybatis類來分析SpringBoot提供的自動配置(AutoConfigure)功能,在此之前首先看一個整合Mybatis的實例。

SpringBoot整合Mybatis

提供SpringBoot整合Mybatis的實例,通過Mybatis實現(xiàn)簡單的增刪改查功能;

1.表數(shù)據(jù)

CREATE TABLE `role` ( `note` varchar(255) CHARACTER SET utf8 DEFAULT NULL, `role_name` varchar(255) DEFAULT NULL, `id` bigint(20) DEFAULT NULL) ENGINE=InnoDB DEFAULT CHARSET=utf8

提供創(chuàng)建role表相關(guān)的sql,對表進行增刪改查操作;

2.整合Mybatis的依賴

主要是mybatis-spring-boot-starter和使用的MySQL驅(qū)動:

<dependency>  <groupId>org.mybatis.spring.boot</groupId>  <artifactId>mybatis-spring-boot-starter</artifactId>  <version>2.0.1</version></dependency><dependency>  <groupId>mysql</groupId>  <artifactId>mysql-connector-java</artifactId>  <version>5.1.29</version></dependency>

3.配置application.properties

提供連接mysql相關(guān)的信息:url,驅(qū)動,用戶名,密碼;

spring.datasource.url=jdbc:mysql://localhost/mybatisspring.datasource.username=rootspring.datasource.password=rootspring.datasource.driver-class-name=com.mysql.jdbc.Driver

4.提供bean和Dao

分別提供表對應(yīng)的bean類和操作數(shù)據(jù)庫的dao類;

public class Role {  private long id;  private String roleName;  private String note;  //省略get/set方法}

@Mapperpublic interface RoleDao {  @Select("SELECT id,role_name as roleName,note FROM role WHERE id = #{id}")  Role findRoleById(@Param("id") long id);}

5.提供Service和Controller

public interface RoleService {  public Role findRoleById(long roleId);}@Servicepublic class RoleServiceImpl implements RoleService {  @Autowired  private RoleDao roleDao;  @Override  public Role findRoleById(long roleId) {    return roleDao.findRoleById(roleId);  }}

@RestControllerpublic class RoleController {  @Autowired  private RoleService roleService;  @RequestMapping("/role")  public String getRole(long id) {    return roleService.findRoleById(id).toString();  }}

啟動服務(wù),進行簡單的測試:http://localhost:8888/role?id=111

結(jié)果如下:

Role [id=111, roleName=zhaohui, note=hello]

6.提出問題

如上實例中,我們使用了很少的配置,就通過mybatis實現(xiàn)了操作數(shù)據(jù)庫;正常使用mybatis需要的SqlSessionFactory和SqlSession沒有看到被實例化,同時mybatis依賴的數(shù)據(jù)源也沒有看到被引用,那SpringBoot是如何幫我們自動配置的,下面重點分析一下;

SpringBoot自動配置

1.自動配置注解

要想使用自動配置功能,SpringBoot提供了注解@EnableAutoConfiguration,當然不需要我們配置因為在@SpringBootApplication注解中默認以及啟用了;

@Target(ElementType.TYPE)@Retention(RetentionPolicy.RUNTIME)@Documented@Inherited@SpringBootConfiguration@EnableAutoConfiguration@ComponentScan(excludeFilters = { @Filter(type = FilterType.CUSTOM, classes = TypeExcludeFilter.class),    @Filter(type = FilterType.CUSTOM, classes = AutoConfigurationExcludeFilter.class) })public @interface SpringBootApplication {//...省略...}

可以看到@SpringBootApplication注解本身也有注解@EnableAutoConfiguration:

@Target(ElementType.TYPE)@Retention(RetentionPolicy.RUNTIME)@Documented@Inherited@AutoConfigurationPackage@Import(AutoConfigurationImportSelector.class)public @interface EnableAutoConfiguration {//...省略...}

在注解@EnableAutoConfiguration中重點看一下@Import注解中使用的AutoConfigurationImportSelector類,此類是自動注解的核心類,會有條件的加載我們默認指定的配置類;這里有兩個概念一個是有條件,一個是配置類,分別簡單介紹一下:配置類可以簡單理解就是相關(guān)組件對接SpringBoot的對接類,此類可以做一些初始化的工作;有條件表示并不是有配置類就能被對接上,是有條件的,SpringBoot默認提供了大量配置類,但并不是所有配置類都能被加載初始化的,是有條件的,比如mybatis在沒有數(shù)據(jù)源的情況下,沒有mybatis基礎(chǔ)包的情況下是不能被對接的;下面首先看一下SpringBoot提供的哪些條件類;

2.條件類

SpringBoot提供了很多條件類,可以在配置中上配置注解條件類,相關(guān)條件類可以在spring-boot-autoconfigure包下的org.springframework.boot.autoconfigure.condition下找到,主要包含如下:

ConditionalOnBean:當前容器有指定Bean的條件下;  ConditionalOnClass:當前類路徑下有指定類的條件下;  ConditionalOnCloudPlatform:當指定了云平臺的時候;  ConditionalOnExpression:SpEL表達式作為判斷條件;  ConditionalOnJava:JVM版本作為判斷條件;  ConditionalOnJndi:在JNDI存在的條件下查找指定的位置;  ConditionalOnMissingBean:當容器里沒有指定Bean的情況下;  ConditionalOnMissingClass:當類路徑下沒有指定的類的條件下;  ConditionalOnNotWebApplication:當前項目不是WEB項目的條件下;  ConditionalOnProperty:當前應(yīng)用是否配置了指定屬性指定的值;  ConditionalOnResource:只有當指定的資源位于類路徑下;  ConditionalOnSingleCandidate:bean工廠中只有一個或者有多個情況下是主要的候選bean;  ConditionalOnWebApplication:當前項目是WEB項目的條件下。

以上是注解類,注解本身沒有功能,只是提供標記的功能,具體功能在@Conditional中指定的,比如ConditionalOnBean注解如下所示:

@Target({ ElementType.TYPE, ElementType.METHOD })@Retention(RetentionPolicy.RUNTIME)@Documented@Conditional(OnBeanCondition.class)public @interface ConditionalOnBean {//...省略...}

相關(guān)功能的實現(xiàn)就在OnBeanCondition類中,同樣其他注解類的實現(xiàn)類也在包org.springframework.boot.autoconfigure.condition下找到;

3.自動配置過程

Springboot應(yīng)用啟動過程中使用ConfigurationClassParser分析配置類,此類中有一個processImports方法,此方法用來處理@Import注解,在@EnableAutoConfiguration注解存在@Import注解,這時候會實例化注解中的AutoConfigurationImportSelector,在其內(nèi)部有一個AutoConfigurationGroup內(nèi)部類,內(nèi)部類有兩個核心方法分別是:process和selectImports;

@Override    public void process(AnnotationMetadata annotationMetadata, DeferredImportSelector deferredImportSelector) {      Assert.state(deferredImportSelector instanceof AutoConfigurationImportSelector,          () -> String.format("Only %s implementations are supported, got %s",              AutoConfigurationImportSelector.class.getSimpleName(),              deferredImportSelector.getClass().getName()));      AutoConfigurationEntry autoConfigurationEntry = ((AutoConfigurationImportSelector) deferredImportSelector)          .getAutoConfigurationEntry(getAutoConfigurationMetadata(), annotationMetadata);      this.autoConfigurationEntries.add(autoConfigurationEntry);      for (String importClassName : autoConfigurationEntry.getConfigurations()) {        this.entries.putIfAbsent(importClassName, annotationMetadata);      }    }

此方法主要獲取經(jīng)過條件過濾之后可用的自動配置類,主要調(diào)用AutoConfigurationImportSelector中的getAutoConfigurationEntry完成的:

protected AutoConfigurationEntry getAutoConfigurationEntry(AutoConfigurationMetadata autoConfigurationMetadata,      AnnotationMetadata annotationMetadata) {    if (!isEnabled(annotationMetadata)) {      return EMPTY_ENTRY;    }    AnnotationAttributes attributes = getAttributes(annotationMetadata);    List<String> configurations = getCandidateConfigurations(annotationMetadata, attributes);    configurations = removeDuplicates(configurations);    Set<String> exclusions = getExclusions(annotationMetadata, attributes);    checkExcludedClasses(configurations, exclusions);    configurations.removeAll(exclusions);    configurations = filter(configurations, autoConfigurationMetadata);    fireAutoConfigurationImportEvents(configurations, exclusions);    return new AutoConfigurationEntry(configurations, exclusions);  }

首先獲取了所有備選的自動配置類,然后刪除了重復(fù)和被排除的類,最后通過條件進行篩選出可用的配置類,下面分別看一下,首先看一下如何獲取所有備選的配置類:

protected List<String> getCandidateConfigurations(AnnotationMetadata metadata, AnnotationAttributes attributes) {    List<String> configurations = SpringFactoriesLoader.loadFactoryNames(getSpringFactoriesLoaderFactoryClass(),        getBeanClassLoader());    Assert.notEmpty(configurations, "No auto configuration classes found in META-INF/spring.factories. If you "        + "are using a custom packaging, make sure that file is correct.");    return configurations;  }

通過SpringFactoriesLoader獲取類路徑下META-INF/spring.factories文件中key為org.springframework.boot.autoconfigure.EnableAutoConfiguration的配置類,可以看一下spring-boot-autoconfigure.jar中的spring.factories內(nèi)容:

# Auto Configureorg.springframework.boot.autoconfigure.EnableAutoConfiguration=\org.springframework.boot.autoconfigure.admin.SpringApplicationAdminJmxAutoConfiguration,\org.springframework.boot.autoconfigure.aop.AopAutoConfiguration,\org.springframework.boot.autoconfigure.amqp.RabbitAutoConfiguration,\org.springframework.boot.autoconfigure.batch.BatchAutoConfiguration,\org.springframework.boot.autoconfigure.cache.CacheAutoConfiguration,\org.springframework.boot.autoconfigure.cassandra.CassandraAutoConfiguration,\org.springframework.boot.autoconfigure.cloud.CloudServiceConnectorsAutoConfiguration,\//...以下省略...

當然這里只是截取了其中一個類路徑j(luò)ar下的部分配置,獲取所有配置類之后進行去重,去被排除的類,然后進行條件過濾,下面重點看一下:

private List<String> filter(List<String> configurations, AutoConfigurationMetadata autoConfigurationMetadata) {    long startTime = System.nanoTime();    String[] candidates = StringUtils.toStringArray(configurations);    boolean[] skip = new boolean[candidates.length];    boolean skipped = false;    for (AutoConfigurationImportFilter filter : getAutoConfigurationImportFilters()) {      invokeAwareMethods(filter);      boolean[] match = filter.match(candidates, autoConfigurationMetadata);      for (int i = 0; i < match.length; i++) {        if (!match[i]) {          skip[i] = true;          candidates[i] = null;          skipped = true;        }      }    }    if (!skipped) {      return configurations;    }    List<String> result = new ArrayList<>(candidates.length);    for (int i = 0; i < candidates.length; i++) {      if (!skip[i]) {        result.add(candidates[i]);      }    }    if (logger.isTraceEnabled()) {      int numberFiltered = configurations.size() - result.size();      logger.trace("Filtered " + numberFiltered + " auto configuration class in "          + TimeUnit.NANOSECONDS.toMillis(System.nanoTime() - startTime) + " ms");    }    return new ArrayList<>(result);  }

此方法大致就是首先獲取配置的AutoConfigurationImportFilter ,然后對之前獲取的所有配置類進行過濾,最后返回過濾之后的配置類;AutoConfigurationImportFilter同樣也是通過SpringFactoriesLoader類進行加載類路徑下META-INF/spring.factories,只不過當前的key是:org.springframework.boot.autoconfigure.AutoConfigurationImportFilter,可以看一下SpringBoot默認配置的filter:

# Auto Configuration Import Filtersorg.springframework.boot.autoconfigure.AutoConfigurationImportFilter=\org.springframework.boot.autoconfigure.condition.OnBeanCondition,\org.springframework.boot.autoconfigure.condition.OnClassCondition,\org.springframework.boot.autoconfigure.condition.OnWebApplicationCondition

可以看到Filter其實就是上文介紹的條件類,這里默認了OnBeanCondition,OnClassCondition以及OnWebApplicationCondition,已這里使用的Mybatis為例看一下MybatisAutoConfiguration的注解:

@org.springframework.context.annotation.Configuration@ConditionalOnClass({ SqlSessionFactory.class, SqlSessionFactoryBean.class })@ConditionalOnSingleCandidate(DataSource.class)@EnableConfigurationProperties(MybatisProperties.class)@AutoConfigureAfter(DataSourceAutoConfiguration.class)public class MybatisAutoConfiguration implements InitializingBean {//...以下省略...}

可以看到其中有用到@ConditionalOnClass,表示必須提供SqlSessionFactory和SqlSessionFactoryBean類的情況下才加載此配置類,而整兩個是正式Mybatis基礎(chǔ)包中提供的;有了基礎(chǔ)包還不行,還需要DataSource,而且DataSource必須在MybatisAutoConfiguration實例化之前初始化好,SpringBoot是如何實現(xiàn),繼續(xù)看另外一個核心方法selectImports():

@Override    public Iterable<Entry> selectImports() {      if (this.autoConfigurationEntries.isEmpty()) {        return Collections.emptyList();      }      Set<String> allExclusions = this.autoConfigurationEntries.stream()          .map(AutoConfigurationEntry::getExclusions).flatMap(Collection::stream).collect(Collectors.toSet());      Set<String> processedConfigurations = this.autoConfigurationEntries.stream()          .map(AutoConfigurationEntry::getConfigurations).flatMap(Collection::stream)          .collect(Collectors.toCollection(LinkedHashSet::new));      processedConfigurations.removeAll(allExclusions);      return sortAutoConfigurations(processedConfigurations, getAutoConfigurationMetadata()).stream()          .map((importClassName) -> new Entry(this.entries.get(importClassName), importClassName))          .collect(Collectors.toList());    }    private List<String> sortAutoConfigurations(Set<String> configurations,        AutoConfigurationMetadata autoConfigurationMetadata) {      return new AutoConfigurationSorter(getMetadataReaderFactory(), autoConfigurationMetadata)          .getInPriorityOrder(configurations);    }

首先是對被排除類的一個過濾,然后接下來重點看一下對配置類進行排序的一個方法,具體操作在類AutoConfigurationSorter中進行的,具體方法為getInPriorityOrder():

public List<String> getInPriorityOrder(Collection<String> classNames) {    AutoConfigurationClasses classes = new AutoConfigurationClasses(this.metadataReaderFactory,        this.autoConfigurationMetadata, classNames);    List<String> orderedClassNames = new ArrayList<>(classNames);    // Initially sort alphabetically    Collections.sort(orderedClassNames);    // Then sort by order    orderedClassNames.sort((o1, o2) -> {      int i1 = classes.get(o1).getOrder();      int i2 = classes.get(o2).getOrder();      return Integer.compare(i1, i2);    });    // Then respect @AutoConfigureBefore @AutoConfigureAfter    orderedClassNames = sortByAnnotation(classes, orderedClassNames);    return orderedClassNames;  }

首先使用order進行排序,然后使用@AutoConfigureBefore和@AutoConfigureAfter就行排序;order其實就是通過注解@AutoConfigureOrder進行排序的,值是一個整數(shù),結(jié)構(gòu)類似如下:

@AutoConfigureOrder(Ordered.HIGHEST_PRECEDENCE + 10)

@AutoConfigureBefore和@AutoConfigureAfter字面意思也很好理解,指定在其他配置類之前和之后,所以可以看到在MybatisAutoConfiguration中有如下配置:

@AutoConfigureAfter(DataSourceAutoConfiguration.class)

表示在DataSourceAutoConfiguration配置類加載之后才會加載Mybatis配置類,這樣就解決了依賴關(guān)系;還有上文提到的Mybatis操作數(shù)據(jù)庫依賴的SqlSessionFactory和SqlSession,都在MybatisAutoConfiguration進行了初始化操作;SpringBoot本身其實以及提供了大量常用組件的自動配置類,我們只需要提供滿足的特定條件,SpringBoot自動會幫我加載初始化等操作,但是肯定也有自定義配置類的需求,下面用一個簡單的實例來看看如何自定義一個自動配置類;

自定義配置類

接下來我們用很簡單的實例來看一下自定義的流程,一個格式化大寫消息的實例;

1.pom文件引入依賴

<groupId>com.format</groupId><artifactId>format-spring-boot-starter</artifactId><version>0.0.1-SNAPSHOT</version><packaging>jar</packaging><name>format-spring-boot-starter</name><url>http://maven.apache.org</url><properties>  <java.version>1.8</java.version>  <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding></properties><dependencies>  <dependency>    <groupId>org.springframework.boot</groupId>    <artifactId>spring-boot-autoconfigure</artifactId>  </dependency></dependencies><dependencyManagement>  <dependencies>    <dependency>      <!-- Import dependency management from Spring Boot -->      <groupId>org.springframework.boot</groupId>      <artifactId>spring-boot-dependencies</artifactId>      <version>1.5.2.RELEASE</version>      <type>pom</type>      <scope>import</scope>    </dependency>  </dependencies></dependencyManagement>

Spring 官方 Starter通常命名為spring-boot-starter-{name}如 spring-boot-starter-web,Spring官方建議非官方Starter命名應(yīng)遵循{name}-spring-boot-starter的格式;

2.服務(wù)類和屬性配置類

@ConfigurationProperties("format.service")public class FormatServiceProperties {  private String type;  //...get/set省略...}public class FormatService {  private String type;  public FormatService(String type) {    this.type = type;  }  public String wrap(String word) {    if(type.equalsIgnoreCase("Upper")){//大寫      return word.toUpperCase();    }else if(type.equalsIgnoreCase("Lower")){//小寫      return word.toLowerCase();    }    return word;  }}

屬性類提供了type參數(shù)可以在application.properties中配置,可配置值包括:upper,lower;

3.自動配置類和創(chuàng)建spring.factories文件

@Configuration@ConditionalOnClass(FormatService.class)@EnableConfigurationProperties(FormatServiceProperties.class)public class FormatAutoConfigure {  @Autowired  private FormatServiceProperties properties;  @Bean  @ConditionalOnMissingBean  FormatService formatService() {    return new FormatService(properties.getType());  }}

這個就是自定義的自動配置類,SpringBoot啟動的時候會根據(jù)條件自動初始化;最后在resources/META-INF/下創(chuàng)建spring.factories文件:

org.springframework.boot.autoconfigure.EnableAutoConfiguration=com.format.FormatAutoConfigure

4.測試

在其他SpringBoot中可以引入上面創(chuàng)建的項目,引入方式也很簡單:

<dependency>  <groupId>com.format</groupId>  <artifactId>format-spring-boot-starter</artifactId>  <version>0.0.1-SNAPSHOT</version></dependency>

同時在application.properties配置格式化類型:

format.service.type=upper

啟動應(yīng)用,瀏覽器訪問http://localhost:8888/format?word=hello,結(jié)果為:HELLO

本文從使用SpringBoot整合Mybatis開始,然后提出使用中產(chǎn)生的疑問,進而通過分析源碼的方式來理解SpringBoot的自動配置機制,最后自定義了一個自動配置類來看看具體如何使用;SpringBoot通過自動配置的方式幫助開發(fā)者減少了很大的工作量,達到開箱即用的效果;但是另一方面如果出現(xiàn)問題需要調(diào)試可能不是那么好定位。

上述就是小編為大家分享的怎樣通過SpringBoot整合Mybatis分析自動配置了,如果剛好有類似的疑惑,不妨參照上述分析進行理解。如果想知道更多相關(guān)知識,歡迎關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道。

網(wǎng)頁名稱:怎樣通過SpringBoot整合Mybatis分析自動配置
文章位置:http://bm7419.com/article44/jdsehe.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供虛擬主機、商城網(wǎng)站、網(wǎng)站制作外貿(mào)網(wǎng)站建設(shè)、企業(yè)網(wǎng)站制作、品牌網(wǎng)站建設(shè)

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時需注明來源: 創(chuàng)新互聯(lián)

手機網(wǎng)站建設(shè)