MyBatis入门及深入完整版

4 篇文章 0 订阅
1 篇文章 0 订阅


要系统的学习还需不断的阅读官方文档,同时这篇文章并不是对官网的照搬,而是结合官网与多个权威教学视频的笔记,以及很多博客和自己的经验总结的文字
mybatis官网

Mybatis

在这里插入图片描述

简单来说可以把Mybatis看作对JDBC的一个封装,且封装的十分优秀。

配置关系
在这里插入图片描述

1. 使用IDEA2020的问题

写入依赖关系的时候,在IDEA2020中,需要手动ctrl+shif+o导入依赖包,核心的依赖只有以下两个

<dependency>
    <groupId>org.mybatis</groupId>
    <artifactId>mybatis</artifactId>
    <version>3.5.5</version>
</dependency>
<dependency>
    <groupId>mysql</groupId>
    <artifactId>mysql-connector-java</artifactId>
    <version>8.0.12</version>
</dependency>

同时,由于mysql版本过高,需要导入较高版本的mysql-connector-java,这就需要手动导入,然后每一次构建依赖关系就ctrl+shift+o导入即可,否则会出现红色报错。

数据库基本信息写法,高版本mysql对useSSL,timezone等都有要求,且驱动在cj.jdbc.Driver。

<!--                连接数据库的基本信息-->
                <property name="driver" value="com.mysql.cj.jdbc.Driver"></property>
                <property name="url" value="jdbc:mysql://localhost:3306/test?characterEncoding=utf-8&amp;useSSL=false&amp;serverTimezone=UTC"></property>
                <property name="username" value="root"></property>
                <property name="password" value="123456"></property>

2.xml配置CRUD(增删改查)

在这里插入图片描述
我们基本的项目结构,我们通过IUserDao接口,IUserDao.xml以及SqlMapConfig.xml实现增删改查,避免了以往创建实现类的操作

  • IUserDao接口用于编写需求
  • SqlMapConfig.xml用于填写连接信息与映射
  • IUserDao.xml类似于以往的Impl实现类,用于编写sql语句

IUserDao.xml

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE mapper
        PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
        "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.czy.dao.DAO">
    <select id="select"  resultType="com.czy.pojo.User">
        select * from user
    </select>

</mapper>

SqlMapConfig.xml

<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE configuration
        PUBLIC "-//mybatis.org//DTD Config 3.0//EN"
        "http://mybatis.org/dtd/mybatis-3-config.dtd">

<configuration>
    <properties resource="db.properties"/>

    <environments default="development">
        <environment id="development">
            <transactionManager type="JDBC"/>
            <dataSource type="POOLED">
                <property name="driver" value="${driver}"/>
                <property name="url" value="${url}"/>
                <property name="username" value="${username}"/>
                <property name="password" value="${password}"/>
            </dataSource>
        </environment>
    </environments>
    <mappers>
        <mapper resource="DAOMapper.xml"/>
    </mappers>
</configuration>

mybatisutil

创建util类是为了避免大量的重复代码,通过mybatisutils的getSqlSession()方法获得sqlSession可以大幅度降低代码冗余

/**
 * @ClassName MyBatisUtils
 * @Description 一个固定的工具类
 * @Author SakuraA6
 * @Date 2020/8/3 9:38
 **/
public class MyBatisUtils {

    private  static SqlSessionFactory factory;
    static {
        try {
            String resource = "SqlMapConfig.xml";
            //  MyBatis 包含一个名叫 Resources 的工具类,它包含一些实用方法,使得从类路径或其它位置加载资源文件更加容易。
            InputStream inputStream = Resources.getResourceAsStream(resource);
            SqlSessionFactoryBuilder builder = new SqlSessionFactoryBuilder();
            factory = builder.build(inputStream);
        } catch (IOException e) {
            e.printStackTrace();
        }

    }

    public static SqlSession getSqlSession(){
        SqlSession sqlSession = factory.openSession();
        return sqlSession;
    }
}

实现类

创建步骤,这个图我们在生命周期里也有板书,首先我们需要一个SqlSessionFactoryBuilder,创建一个带有SqlSessionConfig.xml配置文件的工厂,通过工厂openSession()获得一个sqlSession。
在这里插入图片描述

public class Test {
    public static void main(String[] args) {
        SqlSession sqlSession = MybatisUtils.getSqlSession();

        DAO dao = sqlSession.getMapper(DAO.class);
        List<User> userList = dao.select();

        for(User user: userList){
            System.out.println(user);
        }
    }
}

输出

com.czy.pojo.User{id=null, name='czy', password='123456'}
com.czy.pojo.User{id=null, name='xiaowang', password='asdfg'}
com.czy.pojo.User{id=null, name='xiaolvlv', password='456233'}
com.czy.pojo.User{id=123, name='xiaowang', password='null'}
com.czy.pojo.User{id=123, name='xiaowang', password='null'}
com.czy.pojo.User{id=null, name='xiaowang', password='null'}
com.czy.pojo.User{id=null, name='xiaowang', password='null'}
com.czy.pojo.User{id=null, name='xiaowang', password='null'}
com.czy.pojo.User{id=123, name='xiaowang', password='null'}

3.xml配置

3.1优化config配置

编写一个db.properties配置文件,注意url的书写格式,mysql版本高了以后有很多要求,且中间连接符在xml、java、properties格式都不一样…

driver=com.mysql.cj.jdbc.Driver
url=jdbc:mysql://localhost:3306/test?characterEncoding=utf-8&useSSL=false&serverTimezone=UTC
username=root
password=123456

config文件,这样就实现了动态配置。

<configuration>
    <!--    引入外部配置文件-->
    <properties resource="db.properties"/>
    <!--    配置环境-->
    <environments default="development">
        <!--        配置mysql的环境-->
        <environment id="development">
            <!--            配置事务的类型-->
            <transactionManager type="JDBC"/>
            <!--            配置数据源-->
            <dataSource type="POOLED">
                <!--                连接数据库的基本信息-->
                <property name="driver" value="${driver}"/>
                <property name="url" value="${url}"/>
                <property name="username" value="${username}"/>
                <property name="password" value="${password}"/>
            </dataSource>
        </environment>

    </environments>

3.2类型别名

用于简化resultType中的类名配置在这里插入图片描述
在config中配置

<!--    给实体类起别名-->
    <typeAliases>
        <typeAlias type="com.czy.Demo01.User" alias="User"></typeAlias>
    </typeAliases>

配置之后
在这里插入图片描述
同样也可以扫描一个包

4. environment

<environments default="development">
  <environment id="development">
    <transactionManager type="JDBC">
      <property name="..." value="..."/>
    </transactionManager>
    <dataSource type="POOLED">
      <property name="driver" value="${driver}"/>
      <property name="url" value="${url}"/>
      <property name="username" value="${username}"/>
      <property name="password" value="${password}"/>
    </dataSource>
  </environment>
</environments>

在这里插入图片描述

5.映射器

既然 MyBatis 的行为已经由上述元素配置完了,我们现在就要来定义 SQL 映射语句了。 我们需要告诉 MyBatis 到哪里去找到这些语句。 在自动查找资源方面,Java 并没有提供一个很好的解决方案,所以最好的办法是直接告诉 MyBatis 到哪里去找映射文件。 你可以使用相对于类路径的资源引用,或完全限定资源定位符(包括 file:/// 形式的 URL),或类名和包名等。例如:

  • class
  • package
  • resource
    在这里插入图片描述

6.xml映射文件

MyBatis 的真正强大在于它的语句映射,这是它的魔力所在。由于它的异常强大,映射器的 XML 文件就显得相对简单。如果拿它跟具有相同功能的 JDBC 代码进行对比,你会立即发现省掉了将近 95% 的代码。MyBatis 致力于减少使用成本,让用户能更专注于 SQL 代码。
在这里插入图片描述
属性的别名,在开发过程中,可能会出现实体类的属性和数据库列名不一致的情况,这时传递值就会出现空,可以在

mapper文件中添加resultMap,将实体类的属性名和列明相匹配。

<resultMap id="userResultMap" type="User">
    <result property="adc" column="id"/>
</resultMap>

在这里插入图片描述

6.1select元素的属性

id在命名空间中唯一的标识符,可以被用来引用这条语句。
parameterType将会传入这条语句的参数的类全限定名或别名。这个属性是可选的,因为 MyBatis 可以通过类型处理器(TypeHandler)推断出具体传入语句的参数,默认值为未设置(unset)。
parameterMap用于引用外部 parameterMap 的属性,目前已被废弃。请使用行内参数映射和 parameterType 属性。
resultType期望从这条语句中返回结果的类全限定名或别名。 注意,如果返回的是集合,那应该设置为集合包含的类型,而不是集合本身的类型。 resultType 和 resultMap 之间只能同时使用一个。
resultMap对外部 resultMap 的命名引用。结果映射是 MyBatis 最强大的特性,如果你对其理解透彻,许多复杂的映射问题都能迎刃而解。 resultType 和 resultMap 之间只能同时使用一个。

等等

6.2问题

bean中的属性名和数据库中的column名不一样,sql查询会出现null

解决方案

  1. 起别名 使用as起别名(更改sql语句) (不推荐)
  2. resultMap select的resultMapresultMap的id要一致
    在这里插入图片描述

7.日志工厂

log4j

7.1 log4j的基础配置

依赖

<dependencies>
        <!--        导入log4j-->
        <dependency>
            <groupId>log4j</groupId>
            <artifactId>log4j</artifactId>
            <version>1.2.12</version>
        </dependency>

        <!--        log4j所依赖的包-->
        <dependency>
            <groupId>org.slf4j</groupId>
            <artifactId>slf4j-log4j12</artifactId>
            <version>1.5.6</version>
        </dependency>

log4j

# 日记级别(单个级别) 文件/控制台
log4j.rootLogger=debug, stdout

# Redirect log messages to console
log4j.appender.stdout=org.apache.log4j.ConsoleAppender
log4j.appender.stdout.Target=System.out
log4j.appender.stdout.layout=org.apache.log4j.PatternLayout
log4j.appender.stdout.layout.ConversionPattern=%d{yyyy-MM-dd HH:mm:ss} %-5p %c{1}:%L - %m%n

8. 注解的CRUD

在创建接口的时候,在接口上使用注解,用config文件连接该接口即可使用。

8.1 字符串替换

而是可以只写这样一个方法:

@Select("select * from user where ${column} = #{value}")
User findByColumn(@Param("column") String column, @Param("value") String value);

其中 ${column} 会被直接替换,而 #{value} 会使用 ? 预处理。 这样,就能完成同样的任务。所以不推荐使用${},因为不安全。

8.2 @param

当方法有多个参数,一定要使用@param注解

9.Lombok

可以简化getter setter和tostring,一款插件。有需求可以百度。

10.生命周期

10.1 SqlSessionFactoryBuilder

这个类可以被实例化、使用和丢弃,一旦创建了 SqlSessionFactory,就不再需要它了。 因此 SqlSessionFactoryBuilder 实例的最佳作用域是方法作用域(也就是局部方法变量)。 你可以重用 SqlSessionFactoryBuilder 来创建多个 SqlSessionFactory 实例,但最好还是不要一直保留着它,以保证所有的 XML 解析资源可以被释放给更重要的事情。

10.2 SqlSessionFactory

SqlSessionFactory 一旦被创建就应该在应用的运行期间一直存在,没有任何理由丢弃它或重新创建另一个实例。 使用 SqlSessionFactory 的最佳实践是在应用运行期间不要重复创建多次,多次重建 SqlSessionFactory 被视为一种代码“坏习惯”。因此 SqlSessionFactory 的最佳作用域是应用作用域。 有很多方法可以做到,最简单的就是使用单例模式或者静态单例模式。

10.3 SqlSession

每个线程都应该有它自己的 SqlSession 实例。SqlSession 的实例不是线程安全的,因此是不能被共享的,所以它的最佳的作用域是请求或方法作用域。 绝对不能将 SqlSession 实例的引用放在一个类的静态域,甚至一个类的实例变量也不行。 也绝不能将 SqlSession 实例的引用放在任何类型的托管作用域中,比如 Servlet 框架中的 HttpSession。 如果你现在正在使用一种 Web 框架,考虑将 SqlSession 放在一个和 HTTP 请求相似的作用域中。 换句话说,每次收到 HTTP 请求,就可以打开一个 SqlSession,返回一个响应后,就关闭它。 这个关闭操作很重要,为了确保每次都能执行关闭操作,你应该把这个关闭操作放到 finally 块中。 下面的示例就是一个确保 SqlSession 关闭的标准模式:

在这里插入图片描述

11. 动态SQL

根据不同条件生成不同的sql语句,具体用法有如下四种。

  • if

  • choose (when, otherwise)

  • trim (where, set)

  • foreach

    例如,在xml中配置如下语句

    <select id="findActiveBlogWithTitleLike"
         resultType="Blog">
      SELECT * FROM BLOG
      WHERE state = ‘ACTIVE’
      <if test="title != null">
        AND title like #{title}
      </if>
    </select>
    

12. Mybatis缓存

默认情况下,只启用了本地的会话缓存,它仅仅对一个会话中的数据进行缓存。 要启用全局的二级缓存,只需要在你的 SQL 映射文件中添加一行

<cache/>

12.1全局缓存

在我们的config文件中进行如下配置
在这里插入图片描述

12.2 使用二级缓存

<cache
  eviction="FIFO"
  flushInterval="60000"
  size="512"
  readOnly="true"/>

将如上代码配置到config,从而mapper能够使用该机制。

这个更高级的配置创建了一个 FIFO (先进先出)缓存,每隔 60 秒刷新,最多可以存储结果对象或列表的 512 个引用,而且返回的对象被认为是只读的,因此对它们进行修改可能会在不同线程中的调用者产生冲突。

程序会先查询二级缓存中有没有再查看一级缓存中有没有
在这里插入图片描述

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值