【笔记】MyBatis

MyBatis

1.MyBatis简介

是一个基于Java的持久层框架

  1. 支持定制化SQL、存储过程以及高级映射
  2. 避免了几乎所有的JDBC代码和手动设置参数以及获取结果集
  3. 可以使用简单的XML或注解用于配置和原始映射,将接口和Java的POJO映射成数据库中的记录

2.Mybatis 中#和$的区别?

  1. #将传入的数据都当成一个字符串,会对自动传入的数据加一个双引号。如: order by #user_id#,如果传入的值是111,那么解析成 sql 时的值为 order by “111”
  2. $将传入的数据直接显示生成在 sql 中。如: order by u s e r i d user_id userid,如果传入的值是 111,那么解析成 sql 时的值为order by user_id
  3. #方式能够很大程度防止 sql 注入、$方式无法防止 Sql 注入。
  4. 一般能用#的就别用$

3.Mybatis 的编程步骤是什么样的?

  1. 创建 SqlSessionFactory
  2. 通过 SqlSessionFactory 创建 SqlSession
  3. 通过 sqlsession 执行数据库操作
  4. 调用 session.commit()提交事务
  5. 调用 session.close()关闭会话

4.使用 MyBatis 的 mapper 接口调用时有哪些要求?

  1. Mapper 接口方法名和 mapper.xml 中定义的每个 sql 的 id 相同
  2. Mapper 接口方法的输入参数类型和 mapper.xml 中定义的每个 sql 的 parameterType 的类型相同
  3. Mapper 接口方法的输出参数类型和 mapper.xml 中定义的每个 sql 的 resultType 的类型相同
  4. Mapper.xml 文件中的 namespace 即是 mapper 接口的类路径。

5.Mybatis 中一级缓存与二级缓存?

  1. 一级缓存: 基于 PerpetualCache 的 HashMap 本地缓存,其存储作用域为 Session,当 Session flush 或
    close 之后,该 Session 中的所有 Cache 就将清空。
  2. 二级缓存与一级缓存其机制相同,默认也是采用 PerpetualCache, HashMap 存储,不同在于其存储作用域为
    Mapper(Namespace),并且可自定义存储源,如 Ehcache。作用域为 namespance 是指对该 namespance 对
    应的配置文件中所有的 select 操作结果都缓存,这样不同线程之间就可以共用二级缓存。
    启动二级缓存:在 mapper 配置文件中: 。
    二级缓存可以设置返回的缓存对象策略: 。当 readOnly="true"时,表示二级缓存返
    回给所有调用者同一个缓存对象实例,调用者可以 update 获取的缓存实例,但是这样可能会造成其他调用者出
    现数据不一致的情况(因为所有调用者调用的是同一个实例)。当 readOnly="false"时,返回给调用者的是二级
    缓存总缓存对象的拷贝,即不同调用者获取的是缓存对象不同的实例,这样调用者对各自的缓存对象的修改不会
    影响到其他的调用者,即是安全的,所以默认是 readOnly=“false”;
  3. 对于缓存数据更新机制,当某一个作用域(一级缓存 Session/二级缓存 Namespaces)的进行了 C/U/D 操作后,
    默认该作用域下所有 select 中的缓存将被 clear。

6. MyBatis 在 insert 插入操作时返回主键 ID

数据库为 MySql 时:

1. <insert id="insert" parameterType="com.test.User" keyProperty="userId" useGeneratedKeys="true" >

“keyProperty”表示返回的 id 要保存到对象的那个属性中,“useGeneratedKeys”表示主键 id 为自增长模式。
MySQL 中做以上配置就 OK 了

7.JDBC 编程有哪些不足之处, MyBatis 是如何解决这些问题的?

  1. 数据库链接创建、释放频繁造成系统资源浪费从而影响系统性能,如果使用数据库链接池可解决此问题。
    解决:在 SqlMapConfig.xml 中配置数据链接池,使用连接池管理数据库链接。

  2. Sql 语句写在代码中造成代码不易维护,实际应用 sql 变化的可能较大, sql 变动需要改变 java 代码。
    解决:将 Sql 语句配置在 XXXXmapper.xml 文件中与 java 代码分离。

  3. 向 sql 语句传参数麻烦,因为 sql 语句的 where 条件不一定,可能多也可能少,占位符需要和参数一一对应。
    解决: Mybatis 自动将 java 对象映射至 sql 语句。

  4. 对结果集解析麻烦, sql 变化导致解析代码变化,且解析前需要遍历,如果能将数据库记录封装成 pojo 对象解析比
    较方便

    解决: Mybatis 自动将 sql 执行结果映射至 java 对象。

8.Mybatis 中一级缓存与二级缓存?

  1. 一级缓存:

    • 默认开启
    • HashMap 本地缓存,其存储作用域为 sqlSession,同一次会话期间只要查询过的数据都会保存在当前sqlSession的map中。
    • 当 Session flush 或close 之后,该 Session 中的所有 Cache 就将清空。
  2. 二级缓存

    • 默认也是采用HashMap 存储

    • 需要手动开启和配置,启动二级缓存:在 mapper 配置文件中:

    • 他是基于Namespace级别的缓存,是全局作用域缓存。并且可自定义存储源。作用域为 namespance 是指对该 namespance 对应的配置文件中所有的 select 操作结果都缓存,这样不同线程之间就可以共用二级缓存。

    • 在Session关闭或提交之后才会生效

    • 二级缓存可以设置返回的缓存对象策略: 。

      当 readOnly="true"时,表示二级缓存返回给所有调用者同一个缓存对象实例,调用者可以 update 获取的缓存实例,但是这样可能会造成其他调用者出现数据不一致的情况(因为所有调用者调用的是同一个实例)。

      当 readOnly="false"时,返回给调用者的是二级缓存总缓存对象的拷贝,即不同调用者获取的是缓存对象不同的实例,这样调用者对各自的缓存对象的修改不会影响到其他的调用者,即是安全的,所以默认是 readOnly=“false”;

  3. 对于缓存数据更新机制,当某一个作用域(一级缓存 Session/二级缓存 Namespaces)的进行了 C/U/D 操作后,默认该作用域下所有 select 中的缓存将被clear

9.动态sql

  • 在xml文件内,以标签的形式编写动态sql,完成逻辑判断和动态拼接sql的功能

  • 常用的sql标签:where、if、when、choose、otherwise、foreach、sql(用于抽取可重用的sql片段)

    <sql id="selectSql"></sql>
    <include refid="selectSql"></include>
    

10.插件运行原理,自定义插件

MyBatis仅可以编写ParameterHandler、ResultSetHandler、StatementHandler、Executor这四种接口的插件,MyBatis通过动态代理,为需要拦截的接口生成代理对象以实现接口方法拦截功能,每当执行这4种接口对象的方法时,就会进入拦截方法(InvocationHandler的invoke方法)

实现MyBatis的Interceptor接口,重写intercept方法,给插件编写注解,指定拦截方法

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值