Spring Boot-(3)集成Mybatis

一般情况下,应用都会和数据发生交互,所以这张主要介绍一下如何和数据库发生交互。

常用的集中方式包括:JdbcTemplate、ORM框架如Hibernate/Mybatis、JPA等等。

我也是看着别人的例子照猫画虎,将Mybatis集成了进来。具体的细节,自己也在慢慢的理解中。

一、jar包的引入,用到的比较多:

<!-- 据说很牛的一个数据源 -->
<dependency>
    <groupId>com.zaxxer</groupId>
    <artifactId>HikariCP</artifactId>
</dependency>
<!-- 阿里开源的一个数据源 -->
<dependency>
    <groupId>com.alibaba</groupId>
    <artifactId>druid</artifactId>
    <version>${druid.version}</version>
</dependency>
<!-- 集成 mybatis -->
<dependency>
    <groupId>org.mybatis.spring.boot</groupId>
    <artifactId>mybatis-spring-boot-starter</artifactId>
    <version>${mybatis-spring-boot-starter.version}</version>
</dependency>
<dependency>
    <groupId>com.github.pagehelper</groupId>
    <artifactId>pagehelper-spring-boot-starter</artifactId>
    <version>${pagehelper-spring-boot-starter.version}</version>
</dependency>
<dependency>
    <groupId>org.hibernate</groupId>
    <artifactId>hibernate-validator</artifactId>
</dependency>
<dependency>
    <groupId>io.jsonwebtoken</groupId>
    <artifactId>jjwt</artifactId>
    <version>${jjwt.version}</version>
</dependency>

用的版本:
<druid.version>1.0.31</druid.version>
<mybatis-spring-boot-starter.version>1.3.0</mybatis-spring-boot-starter.version>
<pagehelper-spring-boot-starter.version>1.1.3</pagehelper-spring-boot-starter.version>
<jjwt.version>0.7.0</jjwt.version>

二、在yml文件中配置数据库的相关信息:

app: 
  jdbc_url: jdbc:mysql://127.0.0.1:3306/udpdb?useUnicode=true&characterEncoding=UTF-8&zeroDateTimeBehavior=convertToNull
  jdbc_username: root
  jdbc_password: xxxx
  jdbc_driverClass: com.mysql.jdbc.Driver
  maxPoolPreparedStatementPerConnectionSize: 50
  # 申请连接时执行validationQuery检测连接是否有效,做了这个配置会降低性能
  testOnBorrow: false
  # 归还连接时执行validationQuery检测连接是否有效
  testOnReturn: false
  # 申请连接的时候检测,如果空闲时间大于timeBetweenEvictionRunsMillis,执行validationQuery检测连接是否有效
  testWhileIdle: true
  validationQuery: SELECT 1
  initialSize: 30
  maxActive:  100

三、增加数据源配置类
实际上这里要还有一些事情没有搞的特别清楚,这个配置类不是必须的,可以用Spring Boot自带的数据源配置,但是我们使用的是第三方的数据源,并且使用Druid的时候,可以使用到它自带的监控统计功能,所以需要写这个配置类。

我放在了这个包路径下面:com.spring.lw.Towns.config

3.1 读取配置文件中的配置

package com.spring.lw.Towns.config;

import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Component;

import lombok.Data;

/**
 * 数据库 配置属性
 *
 * @author duyq06
 */
@Component
@Data
public class DbConfigProperties {

  @Value("${app.jdbc_url}")
  private String jdbc_url;

  @Value("${app.jdbc_username}")
  private String jdbc_username;

  @Value("${app.jdbc_password}")
  private String jdbc_password;

  @Value("${app.jdbc_driverClass}")
  private String jdbc_driverClass;

  @Value("${app.maxPoolPreparedStatementPerConnectionSize}")
  private int maxPoolPreparedStatementPerConnectionSize;

  @Value("${app.testOnBorrow}")
  private boolean testOnBorrow;

  @Value("${app.testOnReturn}")
  private boolean testOnReturn;

  @Value("${app.testWhileIdle}")
  private boolean testWhileIdle;

  @Value("${app.validationQuery}")
  private String validationQuery;

  @Value("${app.initialSize}")
  private int initialSize;

  @Value("${app.maxActive}")
  private int maxActive;
}

3.2 具体配置数据源和监控,非生产环境使用的是Druid数据源,生产环境使用效率更高的HikariCP

package com.spring.lw.Towns.config;

import java.util.HashMap;
import java.util.Map;

import javax.sql.DataSource;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.boot.autoconfigure.jdbc.DataSourceBuilder;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.boot.web.servlet.FilterRegistrationBean;
import org.springframework.boot.web.servlet.ServletRegistrationBean;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.Primary;
import org.springframework.context.annotation.Profile;
import org.springframework.jdbc.core.JdbcTemplate;

import com.alibaba.druid.pool.DruidDataSource;
import com.alibaba.druid.support.http.StatViewServlet;
import com.alibaba.druid.support.http.WebStatFilter;
import com.zaxxer.hikari.HikariConfig;
import com.zaxxer.hikari.HikariDataSource;

/** 配置 DataSourceConfig */
@Configuration
public class DataSourceConfig {

  @Autowired private DbConfigProperties dbConfigProperties;
  /**
   * 注册DruidServlet 启用数据源的Web监控统计功能
   *
   * @return
   */
  @Profile({"development", "dev", "int", "uat", "test"})
  @Bean
  public ServletRegistrationBean druidServletRegistrationBean() {
    ServletRegistrationBean servletRegistrationBean = new ServletRegistrationBean();
    servletRegistrationBean.setServlet(new StatViewServlet());
    servletRegistrationBean.addUrlMappings("/druid/*");
    return servletRegistrationBean;
  }

  /**
   * 注册DruidFilter拦截
   *
   * @return
   */
  @Profile({"development", "dev", "int", "uat", "test"})
  @Bean
  public FilterRegistrationBean duridFilterRegistrationBean() {
    FilterRegistrationBean filterRegistrationBean = new FilterRegistrationBean();
    filterRegistrationBean.setFilter(new WebStatFilter());
    Map<String, String> initParams = new HashMap<String, String>();
    // 设置忽略请求
    initParams.put("exclusions", "*.js,*.gif,*.jpg,*.bmp,*.png,*.css,*.ico,/druid/*");
    filterRegistrationBean.setInitParameters(initParams);
    filterRegistrationBean.addUrlPatterns("/*");
    return filterRegistrationBean;
  }

  /**
   * 测试环境可以用 DruidDataSource 以可视化观察应用sql 的执行状况
   *
   * @return
   */
  @Profile({"development", "dev", "int", "uat", "test"})
  @Bean(name = "appDataSource")
  @Primary
  public DataSource appDataSource() {
    DruidDataSource dataSource = new DruidDataSource();
    dataSource.setUrl(this.dbConfigProperties.getJdbc_url());
    dataSource.setUsername(this.dbConfigProperties.getJdbc_username());
    dataSource.setPassword(this.dbConfigProperties.getJdbc_password());
    dataSource.setDriverClassName(this.dbConfigProperties.getJdbc_driverClass());
    dataSource.setTestOnReturn(this.dbConfigProperties.isTestOnReturn());
    dataSource.setTestWhileIdle(this.dbConfigProperties.isTestWhileIdle());
    dataSource.setTestOnBorrow(this.dbConfigProperties.isTestOnBorrow());
    dataSource.setInitialSize(this.dbConfigProperties.getInitialSize());
    dataSource.setMaxPoolPreparedStatementPerConnectionSize(
        this.dbConfigProperties.getMaxPoolPreparedStatementPerConnectionSize());
    dataSource.setMaxActive(this.dbConfigProperties.getMaxActive());
    dataSource.setValidationQuery(this.dbConfigProperties.getValidationQuery());
    return dataSource;
  }

  /**
   * 生产环境选择性能更高的HikariCP https://github.com/brettwooldridge/HikariCP
   *
   * @return
   */
  @Profile({"prd"})
  @Bean(name = "appDataSource")
  @Primary
  public DataSource hikariDataSource() {
    HikariConfig config = new HikariConfig();
    config.setJdbcUrl(this.dbConfigProperties.getJdbc_url());
    config.setUsername(this.dbConfigProperties.getJdbc_username());
    config.setPassword(this.dbConfigProperties.getJdbc_password());
    config.setDriverClassName(this.dbConfigProperties.getJdbc_driverClass());
    config.setConnectionTestQuery(this.dbConfigProperties.getValidationQuery());
    config.setMaximumPoolSize(
        this.dbConfigProperties.getMaxPoolPreparedStatementPerConnectionSize());
    config.addDataSourceProperty("cachePrepStmts", "true");
    config.addDataSourceProperty("prepStmtCacheSize", "250");
    config.addDataSourceProperty("prepStmtCacheSqlLimit", "2048");
    HikariDataSource dataSource = new HikariDataSource(config);
    return dataSource;
  }

  @Bean(name = "appJdbcTemplate")
  public JdbcTemplate primaryJdbcTemplate(@Qualifier("appDataSource") DataSource dataSource) {
    return new JdbcTemplate(dataSource);
  }
}

四、MyBatis的相关配置

4.1

package com.spring.lw.Towns.mybatis.config;

import javax.sql.DataSource;

import org.apache.ibatis.session.SqlSessionFactory;
import org.mybatis.spring.SqlSessionFactoryBean;
import org.mybatis.spring.SqlSessionTemplate;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.boot.autoconfigure.AutoConfigureAfter;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.core.io.support.PathMatchingResourcePatternResolver;
import org.springframework.core.io.support.ResourcePatternResolver;
import org.springframework.jdbc.datasource.DataSourceTransactionManager;
import org.springframework.transaction.PlatformTransactionManager;
import org.springframework.transaction.annotation.TransactionManagementConfigurer;

import com.spring.lw.Towns.config.DataSourceConfig;


@Configuration
@AutoConfigureAfter(DataSourceConfig.class)
public class MyBatisConfig implements TransactionManagementConfigurer {

  @Autowired
  @Qualifier("appDataSource")
  private DataSource appDataSource;

  @Bean(name = "sqlSessionFactory")
  public SqlSessionFactory sqlSessionFactoryBean() {
    SqlSessionFactoryBean bean = new SqlSessionFactoryBean();
    bean.setDataSource(this.appDataSource);
    bean.setTypeAliasesPackage("com.spring.lw.Towns.mybatis.domain");

    //添加XML目录
    ResourcePatternResolver resolver = new PathMatchingResourcePatternResolver();
    try {
      bean.setMapperLocations(
          resolver.getResources("classpath:/com/spring/lw/Towns/mybatis/mapper/*Mapper.xml"));
      return bean.getObject();
    } catch (Exception e) {
      e.printStackTrace();
      throw new RuntimeException(e);
    }
  }

  @Bean
  public SqlSessionTemplate sqlSessionTemplate(SqlSessionFactory sqlSessionFactory) {
    return new SqlSessionTemplate(sqlSessionFactory);
  }

  @Bean
  public PlatformTransactionManager annotationDrivenTransactionManager() {
    return new DataSourceTransactionManager(this.appDataSource);
  }
}

4.2

package com.spring.lw.Towns.mybatis.config;

import org.apache.ibatis.annotations.Mapper;
import org.mybatis.spring.mapper.MapperScannerConfigurer;
import org.springframework.boot.autoconfigure.AutoConfigureAfter;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

@Configuration
@AutoConfigureAfter(MyBatisConfig.class)
public class MyBatisMapperScannerConfig {
  @Bean
  public MapperScannerConfigurer mapperScannerConfigurer() {
    MapperScannerConfigurer mapperScannerConfigurer = new MapperScannerConfigurer();
    mapperScannerConfigurer.setSqlSessionFactoryBeanName("sqlSessionFactory");
    mapperScannerConfigurer.setBasePackage("com.spring.lw.Towns.mybatis.domain");
    mapperScannerConfigurer.setAnnotationClass(Mapper.class);
    return mapperScannerConfigurer;
  }
}

五、Mapper

以往我们都比较习惯将sql文件写在mapper.xml文件中,还有一个方法,可以直接将sql和方法进行绑定。

package com.spring.lw.Towns.mybatis.domain;

import org.apache.ibatis.annotations.Mapper;
import org.apache.ibatis.annotations.Param;
import org.apache.ibatis.annotations.Select;

import com.spring.lw.Towns.vo.User;

@Mapper
public interface UserMapper {
    
    @Select("select * from USER where id = #{id}")
    public User getUserByID(@Param("id")int id);

}

一切OK,但是还有很多不明白的,感觉垮过去一步,先研究一下Spring Boot docs上的例子,补上。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,539评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,911评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,337评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,723评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,795评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,762评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,742评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,508评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,954评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,247评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,404评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,104评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,736评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,352评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,557评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,371评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,292评论 2 352

推荐阅读更多精彩内容