Mybatis中自定義實(shí)例化SqlSessionFactoryBean問(wèn)題
Mybatis自定義實(shí)例化SqlSessionFactoryBean
現(xiàn)在SpringBoot基本成為開發(fā)的標(biāo)配,如果你上司讓你搭建一個(gè)SpringBoot,然后集成Mybatis+Druid,你可以能百度幾下,卡卡就搭建完畢了。
現(xiàn)在項(xiàng)目基本都會(huì)使用連接池技術(shù),市面上的連接池有很多,比如:DBCP、c3p0、Druid等,今天我們重點(diǎn)介紹Druid連接池。
application.yml配置文件如下所示:
spring: ? #數(shù)據(jù)庫(kù)配置 ? datasource: ? ? druid: ? ? ? type: com.alibaba.druid.pool.DruidDataSource ? ? ? url: jdbc:mysql://127.0.0.1:3306/test?useUnicode=true&characterEncoding=UTF-8&zeroDateTimeBehavior=round&serverTimezone=GMT%2B8 ? ? ? username: test ? ? ? password: test ? ? ? driver-class-name: com.mysql.jdbc.Driver # ? ? 獲取連接時(shí)最大等待時(shí)間,單位毫秒 ? ? ? max-wait: 60000 # ? ? ?最大連接池?cái)?shù)量 ? ? ? max-active: 80 # ? ? ?初始化時(shí)建立物理連接的個(gè)數(shù) ? ? ? initial-size: 20 # ? ? ?最小連接池?cái)?shù)量 ? ? ? min-idle: 40 #Destory線程中如果檢測(cè)到當(dāng)前連接的最后活躍時(shí)間和當(dāng)前時(shí)間的差值大于minEvictableIdleTimeMillis,則關(guān)閉當(dāng)前連接。 ? ? ? min-evictable-idle-time-millis: 600000 # ? ? ?testWhileIdle的判斷依據(jù),詳細(xì)看testWhileIdle屬性的說(shuō)明 ? ? ? time-between-eviction-runs-millis: 2000 #用來(lái)檢測(cè)連接是否有效的sql,要求是一個(gè)查詢語(yǔ)句。 ? ? ? validation-query: select 1 # ? ? ?申請(qǐng)連接的時(shí)候檢測(cè),如果空閑時(shí)間大于timeBetweenEvictionRunsMillis,執(zhí)行validationQuery檢測(cè)連接是否有效。 ? ? ? test-while-idle: true # ? ? ?申請(qǐng)連接時(shí)執(zhí)行validationQuery檢測(cè)連接是否有效,做了這個(gè)配置會(huì)降低性能。 ? ? ? test-on-borrow: false # ? ? ?歸還連接時(shí)執(zhí)行validationQuery檢測(cè)連接是否有效,做了這個(gè)配置會(huì)降低性能 ? ? ? test-on-return: false # ? ? ?屬性類型是字符串,通過(guò)別名的方式配置擴(kuò)展插件 ? ? ? filters: stat,wall # ? ? ?開啟慢sql,并設(shè)置時(shí)間 ? ? ? filter: ? ? ? ? stat.log-slow-sql: true ? ? ? ? stat.slow-sql-millis: 2000 ? ? ? web-stat-filter: ? ? ? ? enabled: true ? ? ? ? url-pattern: /* ? ? ? stat-view-servlet: ? ? ? ? enabled: true ? ? ? ? url-pattern: /druid/*
使用SpringBoot作為項(xiàng)目框架自然簡(jiǎn)單,沒(méi)有很多的xml配置文件,不需要配置額外的tomcat,不管是開發(fā)還是部署都非常方便。但高度集成有時(shí)候也會(huì)帶來(lái)一些麻煩。
比如你上司要求你在mybatis中集成插件并可以識(shí)別common模塊的mybatis.xml映射文件。
這個(gè)時(shí)候你可能首先會(huì)百度如何配置mybatis插件、如何配置多模塊的mapper-locations,然后有很多博文會(huì)說(shuō)在SqlSessionFactoryBean設(shè)置即可。
你可能會(huì)這么設(shè)置:
@Autowired private SqlSessionFactoryBean sqlSessionFactoryBean;
但是結(jié)果不是那么盡人意,初始化的結(jié)果為null。
這是因?yàn)楦甙姹镜膍ybatis實(shí)現(xiàn)機(jī)制做了一些修改,我們沒(méi)辦法通過(guò)@Autowired來(lái)實(shí)例化SqlSessionFactoryBean對(duì)象。
所以我們必須自己來(lái)實(shí)例化SqlSessionFactoryBean對(duì)象,而實(shí)例化SqlSessionFactoryBean對(duì)象的關(guān)鍵就是設(shè)置DataSource數(shù)據(jù)源。
我們可以通過(guò)如下代碼來(lái)實(shí)例化過(guò)SqlSessionFactoryBean。
/**
?* mybatis配置
?* @author linzhiqinag
?*/
@Configuration
public class MybatisConfig {
? ? private Logger logger = LoggerFactory.getLogger(MybatisConfig.class);
?
? ? @Value("${mybatis.mapper-locations}")
? ? private String mapperLocation;
?
? ? @Value("${common-mybatis.mapper-locations}")
? ? private String commonMapperLocation;
?
? ? @Value("${spring.datasource.druid.username}")
? ? private String username;
?
? ? @Value("${spring.datasource.druid.password}")
? ? private String password;
?
? ? @Value("${spring.datasource.druid.url}")
? ? private String dbUrl;
?
? ? @Value("${spring.datasource.druid.initial-size}")
? ? private int initialSize;
?
? ? @Value("${spring.datasource.druid.min-idle}")
? ? private int minIdle;
?
? ? @Value("${spring.datasource.druid.max-active}")
? ? private int maxActive;
?
? ? @Value("${spring.datasource.druid.max-wait}")
? ? private long maxWait;
?
? ? @Value("${spring.datasource.druid.driver-class-name}")
? ? private String driverClassName;
?
? ? @Value("${spring.datasource.druid.min-evictable-idle-time-millis}")
? ? private long minEvictableIdleTimeMillis;
?
? ? @Value("${spring.datasource.druid.time-between-eviction-runs-millis}")
? ? private long timeBetweenEvictionRunsMillis;
?
? ? @Value("${spring.datasource.druid.validation-query}")
? ? private String validationQuery;
?
? ? @Value("${spring.datasource.druid.test-while-idle}")
? ? private boolean testWhileIdle;
?
? ? @Value("${spring.datasource.druid.test-on-borrow}")
? ? private boolean testOnBorrow;
?
? ? @Value("${spring.datasource.druid.test-on-return}")
? ? private boolean testOnReturn;
?
? ? @Value("${spring.datasource.druid.filter.stat.log-slow-sql}")
? ? private boolean logSlowSql;
?
? ? @Value("${spring.datasource.druid.filter.stat.slow-sql-millis}")
? ? private long slowSqlMillis;
?
? ? @Bean
? ? public DruidDataSource dataSource() {
? ? ? ? DruidDataSource druidDataSource = new DruidDataSource();
? ? ? ? try {
? ? ? ? ? ? druidDataSource.setUsername(username);
? ? ? ? ? ? druidDataSource.setPassword(password);
? ? ? ? ? ? druidDataSource.setUrl(dbUrl);
? ? ? ? ? ? druidDataSource.setFilters("stat,wall");
? ? ? ? ? ? druidDataSource.setInitialSize(initialSize);
? ? ? ? ? ? druidDataSource.setMinIdle(minIdle);
? ? ? ? ? ? druidDataSource.setMaxActive(maxActive);
? ? ? ? ? ? druidDataSource.setMaxWait(maxWait);
? ? ? ? ? ? druidDataSource.setMinEvictableIdleTimeMillis(minEvictableIdleTimeMillis);
? ? ? ? ? ? druidDataSource.setTimeBetweenEvictionRunsMillis(timeBetweenEvictionRunsMillis);
? ? ? ? ? ? druidDataSource.setUseGlobalDataSourceStat(true);
? ? ? ? ? ? druidDataSource.setDriverClassName(driverClassName);
? ? ? ? ? ? druidDataSource.setValidationQuery(validationQuery);
? ? ? ? ? ? druidDataSource.setTestWhileIdle(testWhileIdle);
? ? ? ? ? ? druidDataSource.setTestOnBorrow(testOnBorrow);
? ? ? ? ? ? druidDataSource.setTestOnReturn(testOnReturn);
? ? ? ? ? ? // 設(shè)置需要的過(guò)濾
? ? ? ? ? ? List<Filter> statFilters =new ArrayList<>();
? ? ? ? ? ? StatFilter statFilter = new StatFilter();
? ? ? ? ? ? statFilter.setLogSlowSql(logSlowSql);
? ? ? ? ? ? statFilter.setSlowSqlMillis(slowSqlMillis);
? ? ? ? ? ? statFilters.add(statFilter);
? ? ? ? ? ? // 設(shè)置慢SQL
? ? ? ? ? ? druidDataSource.setProxyFilters(statFilters);
? ? ? ? ?} catch (Exception e) {
? ? ? ? ? ? e.printStackTrace();
? ? ? ? }
? ? ? ? return druidDataSource;
? ? }
?
? ? @Bean
? ? public SqlSessionFactoryBean mysqlSessionFactory(DataSource dataSource) throws Exception {
? ? ? ? SqlSessionFactoryBean sqlSessionFactoryBean = new SqlSessionFactoryBean();
? ? ? ? sqlSessionFactoryBean.setDataSource(dataSource);
? ? ? ? PathMatchingResourcePatternResolver resolver = new PathMatchingResourcePatternResolver();
? ? ? ? Resource[] resources1 = resolver.getResources(mapperLocation);
? ? ? ? Resource[] resources2 = resolver.getResources(commonMapperLocation);
? ? ? ? Resource[] resources = new Resource[resources1.length+resources2.length];
? ? ? ? for (int i=0;i<resources1.length;i++) {
? ? ? ? ? ? resources[i] = resources1[i];
? ? ? ? }
? ? ? ? int initSize = resources1.length;
? ? ? ? for (int i=0;i<resources2.length;i++) {
? ? ? ? ? ? resources[initSize+i] = resources2[i];
? ? ? ? }
? ? ? ? sqlSessionFactoryBean.setMapperLocations(resources);
? ? ? ? sqlSessionFactoryBean.setPlugins(new Interceptor[]{new CatMybatisInterceptor(dbUrl)});
? ? ? ? return sqlSessionFactoryBean;
? ? }
}這樣我們就可以得到SqlSessionFactoryBean對(duì)象了,然后我們就可以通過(guò)sqlSessionFactoryBean.setMapperLocations()來(lái)設(shè)置多模塊xml映射,通過(guò)sqlSessionFactoryBean.setPlugins()來(lái)設(shè)置指定的插件了。
注意:
這邊需要注意的是,如果采用代碼的方式實(shí)例化SqlSessionFactoryBean,那關(guān)于數(shù)據(jù)庫(kù)相關(guān)的配置將會(huì)失效,所以在設(shè)置數(shù)據(jù)源的時(shí)候一定要設(shè)置全。
MyBatis中SqlSessionFactoryBean的作用
在使用Spring+MyBatis的環(huán)境下,我們需要配值一個(gè)SqlSessionFactoryBean來(lái)充當(dāng)SqlSessionFactory,這里我們要搞清楚的就是為什么SqlSessionFactoryBean為什么能在Spring IoC容器中以SqlSessionFactory的類型保存并被獲取。
我們來(lái)看看SqlSessionFactoryBean的定義是怎樣的:
public class SqlSessionFactoryBean implements FactoryBean<SqlSessionFactory>, InitializingBean, ApplicationListener<ApplicationEvent> {
?
}能被Spring IoC容器管理的原因就是繼承了FactoryBean這個(gè)接口了,這是個(gè)支持泛型的接口:
public interface FactoryBean<T> {
?? ?@Nullable
?? ?T getObject() throws Exception;
?? ?
?? ?@Nullable
?? ?Class<?> getObjectType();
?? ?
?? ?default boolean isSingleton() {
?? ??? ?return true;
?? ?}
}當(dāng)實(shí)現(xiàn)了這個(gè)接口的Bean在配置為被Spring接管時(shí),存入IoC容器中的實(shí)例類型將會(huì)是實(shí)例化泛型的那個(gè)類型,從IoC容器中獲取時(shí)也是實(shí)例化泛型的那個(gè)類型,這種情況下,Spring 將會(huì)在應(yīng)用啟動(dòng)時(shí)為你創(chuàng)建SqlSessionFactory對(duì)象,然后將它以 SqlSessionFactory為名來(lái)存儲(chǔ)。
當(dāng)把這個(gè)bean注入到Spring中去了以后,IoC容器中的其他類型就可以拿到SqlSession實(shí)例了,就可以進(jìn)行相關(guān)的SQL執(zhí)行任務(wù)了。
總結(jié)
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
Intellij IDEA 最全超實(shí)用快捷鍵整理(長(zhǎng)期更新)
這篇文章主要介紹了Intellij IDEA 最全實(shí)用快捷鍵整理(長(zhǎng)期更新),本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-02-02
通過(guò)案例了解靜態(tài)修飾符static使用場(chǎng)景
這篇文章主要介紹了通過(guò)案例了解靜態(tài)修飾符static使用場(chǎng)景,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-10-10
Java實(shí)現(xiàn)TopK問(wèn)題的方法
這篇文章主要介紹了Java實(shí)現(xiàn)TopK問(wèn)題的方法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-08-08
利用Java反射機(jī)制實(shí)現(xiàn)對(duì)象相同字段的復(fù)制操作
這篇文章主要介紹了利用Java反射機(jī)制實(shí)現(xiàn)對(duì)象相同字段的復(fù)制操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-08-08
如何修改nacos權(quán)重報(bào)錯(cuò)問(wèn)題
這篇文章主要介紹了如何修改nacos權(quán)重報(bào)錯(cuò)問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-07-07
Intellij?IDEA創(chuàng)建web項(xiàng)目的超詳細(xì)步驟記錄
如果剛開始接觸IDEA,或者之前使用的是eclipse/myEclipse的話,即使是創(chuàng)建一個(gè)JAVA WEB項(xiàng)目,估計(jì)也讓很多人費(fèi)了好幾個(gè)小時(shí),下面這篇文章主要給大家介紹了關(guān)于Intellij?IDEA創(chuàng)建web項(xiàng)目的超詳細(xì)步驟,需要的朋友可以參考下2022-08-08
Java案例使用集合方法實(shí)現(xiàn)統(tǒng)計(jì)任意字符串中字符出現(xiàn)的次數(shù)
這篇文章主要介紹了Java案例使用集合方法實(shí)現(xiàn)統(tǒng)計(jì)任意字符串中字符出現(xiàn)的次數(shù),下面我們將用兩種方法實(shí)現(xiàn),需要的小伙伴可以參考一下文章具體內(nèi)容2022-04-04

