通过了解数据源,我们已经清楚,Mybatis实际上是在使用自己编写的数据源(数据源有很多,之后我们再聊其他的)默认使用的是池化的数据源,它预先存储了很多的连接对象。
那么我们来看一下,如何将Mybatis与Spring更好的结合呢,比如我们现在希望将SqlSessionFactory交给IoC容器进行管理,而不是我们自己创建工具类来管理(我们之前一直都在使用工具类管理和创建会话)
首先导入依赖:
- <dependency>
- <groupId>mysqlgroupId>
- <artifactId>mysql-connector-javaartifactId>
- <version>8.0.25version>
- dependency>
- <dependency>
- <groupId>org.mybatisgroupId>
- <artifactId>mybatisartifactId>
- <version>3.5.7version>
- dependency>
- <dependency>
- <groupId>org.mybatisgroupId>
- <artifactId>mybatis-springartifactId>
- <version>2.0.6version>
- dependency>
- <dependency>
- <groupId>org.springframeworkgroupId>
- <artifactId>spring-jdbcartifactId>
- <version>5.3.13version>
- dependency>
在mybatis-spring依赖中,为我们提供了SqlSessionTemplate类,它其实就是官方封装的一个工具类,我们可以将其注册为Bean,这样我们随时都可以向IoC容器索要,而不用自己再去编写一个工具类了,我们可以直接在配置类中创建:
- @Configuration
- @ComponentScan("com.test")
- public class TestConfiguration {
- @Bean
- public SqlSessionTemplate sqlSessionTemplate() throws IOException {
- SqlSessionFactory factory = new SqlSessionFactoryBuilder().build(Resources.getResourceAsReader("mybatis-config.xml"));
- return new SqlSessionTemplate(factory);
- }
- }
- "1.0" encoding="UTF-8" ?>
- configuration
- PUBLIC "-//mybatis.org//DTD Config 3.0//EN"
- "http://mybatis.org/dtd/mybatis-3-config.dtd">
- <configuration>
- <environments default="development">
- <environment id="development">
- <transactionManager type="JDBC"/>
- <dataSource type="POOLED">
- <property name="driver" value="com.mysql.cj.jdbc.Driver"/>
- <property name="url" value="jdbc:mysql://localhost:3306/study"/>
- <property name="username" value="root"/>
- <property name="password" value="123456"/>
- dataSource>
- environment>
- environments>
- <mappers>
- <mapper class="com.test.mapper.TestMapper"/>
- mappers>
- configuration>
- public static void main(String[] args) {
- ApplicationContext context = new AnnotationConfigApplicationContext(TestConfiguration.class);
- SqlSessionTemplate template = context.getBean(SqlSessionTemplate.class);
- TestMapper testMapper = template.getMapper(TestMapper.class);
- System.out.println(testMapper.getStudent());
- }
- @Data
- public class Student {
- int sid;
- String name;
- String sex;
- }
最后成功得到Student实体类,证明SqlSessionTemplate成功注册为Bean可以使用了。虽然这样已经很方便了,但是还不够方便,我们依然需要手动去获取Mapper对象,那么能否直接得到对应的Mapper对象呢,我们希望让Spring直接帮助我们管理所有的Mapper,当需要时,可以直接从容器中获取,我们可以直接在配置类上方添加注解:
@MapperScan("com.test.mapper")
这样,Spring会自动扫描所有的Mapper,并将其实现注册为Bean,那么我们现在就可以直接通过容器获取了:
- public static void main(String[] args) throws InterruptedException {
- ApplicationContext context = new AnnotationConfigApplicationContext(TestConfiguration.class);
- TestMapper mapper = context.getBean(TestMapper.class);
- System.out.println(mapper.getStudent());
- }
请一定注意,必须存在SqlSessionTemplate或是SqlSessionFactoryBean的Bean,否则会无法初始化(毕竟要数据库的链接信息)
我们接着来看,如果我们希望直接去除Mybatis的配置文件,那么改怎么去实现呢?我们可以使用SqlSessionFactoryBean类:
- @Configuration
- @ComponentScan("com.test")
- @MapperScan("com.test.mapper")
- public class TestConfiguration {
- @Bean
- public DataSource dataSource(){
- return new PooledDataSource("com.mysql.cj.jdbc.Driver",
- "jdbc:mysql://localhost:3306/study", "root", "123456");
- }
-
- @Bean
- public SqlSessionFactoryBean sqlSessionFactoryBean(@Autowired DataSource dataSource){
- SqlSessionFactoryBean bean = new SqlSessionFactoryBean();
- bean.setDataSource(dataSource);
- return bean;
- }
- }
删除配置文件,重新再来运行,同样可以正常使用Mapper。从这里开始,通过IoC容器,Mybatis已经不再需要使用配置文件了,之后基于Spring的开发将不会再出现Mybatis的配置文件。
使用HikariCP连接池
前面我们提到了数据源还有其他实现,比如C3P0、Druid等,它们都是非常优秀的数据源实现(可以自行了解),不过我们这里要介绍的,是之后在SpringBoot中还会遇到的HikariCP连接池。
HikariCP是由日本程序员开源的一个数据库连接池组件,代码非常轻量,并且速度非常的快。根据官方提供的数据,在酷睿i7开启32个线程32个连接的情况下,进行随机数据库读写操作,HikariCP的速度是现在常用的C3P0数据库连接池的数百倍。在SpringBoot2.0中,官方也是推荐使用HikariCP。
首先,我们需要导入依赖:
- <dependency>
- <groupId>com.zaxxergroupId>
- <artifactId>HikariCPartifactId>
- <version>3.4.5version>
- dependency>
接着修改一下Bean的定义:
- @Bean
- public DataSource dataSource() throws SQLException {
- HikariDataSource dataSource = new HikariDataSource();
- dataSource.setJdbcUrl("jdbc:mysql://localhost:3306/study");
- dataSource.setDriverClassName("com.mysql.cj.jdbc.Driver");
- dataSource.setUsername("root");
- dataSource.setPassword("123456");
- return dataSource;
- }
最后我们发现,同样可以得到输出结果,但是出现了一个报错:
- SLF4J: Failed to load class "org.slf4j.impl.StaticLoggerBinder".
- SLF4J: Defaulting to no-operation (NOP) logger implementation
- SLF4J: See http://www.slf4j.org/codes.html#StaticLoggerBinder for further details.
此数据源实际上是采用了SLF4J日志框架打印日志信息,但是现在没有任何的日志实现(slf4j只是一个API标准,它规范了多种日志框架的操作,统一使用SLF4J定义的方法来操作不同的日志框架)我们这里就使用JUL作为日志实现,我们需要导入另一个依赖:
- <dependency>
- <groupId>org.slf4jgroupId>
- <artifactId>slf4j-jdk14artifactId>
- <version>1.7.25version>
- dependency>
注意版本一定要和slf4j-api保持一致!
这样就能得到我们的日志信息了:
- 十二月 07, 2021 8:46:41 下午 com.zaxxer.hikari.HikariDataSource getConnection
- 信息: HikariPool-1 - Starting...
- 十二月 07, 2021 8:46:41 下午 com.zaxxer.hikari.HikariDataSource getConnection
- 信息: HikariPool-1 - Start completed.
- Student(sid=1, name=小明, sex=男)
在SpringBoot阶段,我们还会遇到HikariPool-1 - Starting...和HikariPool-1 - Start completed.同款日志信息。
当然,Lombok肯定也是支持这个日志框架快速注解的:
- @Slf4j
- public class Main {
- public static void main(String[] args) {
- log.info("项目正在启动...");
- ApplicationContext context = new AnnotationConfigApplicationContext(TestConfiguration.class);
- TestMapper mapper = context.getBean(TestMapper.class);
- System.out.println(mapper.getStudent());
- }
- }