Mybatis面试题

1,什么是Mybatis

1  Mybatis 是一个半 ORM(对象关系映射)框架,它内部封装了 JDBC,开发时

只需要关注 SQL 语句本身,不需要花费精力去处理加载驱动、创建连接、创建

statement 等繁杂的过程。程序员直接编写原生态 sql,可以严格控制 sql 执行性

能,灵活度高。

2  MyBatis 可以使用 XML 或注解来配置和映射原生信息,将 POJO 映射成数

据库中的记录,避免了几乎所有的 JDBC 代码和手动设置参数以及获取结果集。

通过 xml 文件或注解的方式将要执行的各种 statement 配置起来,并通过

 java 对象和 statement 中 sql 的动态参数进行映射生成最终执行的 sql 语句,最

后由 mybatis 框架执行 sql 并将结果映射为 java 对象并返回。(从执行 sql 到返

回 result 的过程)。

2,Mybatis的优点

1、简单易学。mybatis本身就很小且简单。没有任何第三方依赖,最简单安装只要两个jar加配置几个sql映射文件,易于学习,易于使用,通过文档和源代码,可以比较完全的掌握它的设计思路和实现;

2、灵活。mybatis不会对应用程序或者数据库的现有设计强加任何影响。 sql写在xml里,便于统一管理和优化。通过sql基本上可以实现我们不使用数据访问框架可以实现的所有功能,或许更多;

3、解除sql与程序代码的耦合。通过提供DAO层,将业务逻辑和数据访问逻辑分离,使系统的设计更清晰,更易维护,更易单元测试。sql和代码的分离,提高了可维护性;

4、提供映射标签,支持对象与数据库的orm字段关系映射;

5、提供对象关系映射标签,支持对象关系组建维护;

6、提供xml标签,支持编写动态sql。

3,Mybatis的缺点

SQL 语句的编写工作量较大,尤其当字段多、关联表多时,对开发人员编写

SQL 语句的功底有一定要求。

SQL 语句依赖于数据库,导致数据库移植性差,不能随意更换数据库。

 

4,MyBatis框架适用场合:

(1)MyBatis专注于SQL本身,是一个足够灵活的DAO层解决方案。

(2)对性能的要求很高,或者需求变化较多的项目,如互联网项目,MyBatis将是不错的选择。

 

5,#{}和${}的区别是什么?

 

1. #将传入的数据都当成一个字符串,会对自动传入的数据加一个双引号。如:order by #user_id#,如果传入的值是111,那么解析成sql时的值为order by "111", 如果传入的值是id,则解析成的sql为order by "id".

   $将传入的数据直接显示生成在sql中。如:order by $user_id$,如果传入的值是111,那么解析成sql时的值为order by 111, 如果传入的值是id,则解析成的sql为order by id.

 

2. #方式能够很大程度防止sql注入。 

    $方式无法防止Sql注入。

Mybatis在处理#{}时,会将sql中的#{}替换为?号,调用PreparedStatement的set方法来赋值;Mybatis在处理${}时,就是把${}替换成变量的值。

select * from emp where empno = 7360 or 1=1  -- ${}

select * from emp where empno = '7369'     -- #{}

没有特殊需求情况下,都要使用#{}

6,Mybatis执行流程

      

 

 

 

1、 mybatis配置

SqlMapConfig.xml,此文件作为mybatis的全局配置文件,配置了mybatis的运行环境等信息。

mapper.xml文件即sql映射文件,文件中配置了操作数据库的sql语句。此文件需要在SqlMapConfig.xml中加载。

2、 通过mybatis环境等配置信息构造SqlSessionFactory即会话工厂

3、 由会话工厂创建sqlSession即会话,操作数据库需要通过sqlSession进行。

4、 mybatis底层自定义了Executor执行器接口操作数据库,Executor接口有两个实现,一个是基本执行器、一个是缓存执行器。

5、 MappedStatement也是mybatis一个底层封装对象,它包装了mybatis配置信息及sql映射信息等。mapper.xml文件中一个sql对应一个Mapped Statement对象,sql的id即是Mapped statement的id。

6、 MappedStatement对sql执行输入参数进行定义,包括HashMap、基本类型、pojo,Executor通过Mapped Statement在执行sql前将输入的java对象映射至sql中,输入参数映射就是jdbc编程中对preparedStatement设置参数。

7、 MappedStatement对sql执行输出结果进行定义,包括HashMap、基本类型、pojo,Executor通过Mapped Statement在执行sql后将输出结果映射至java对象中,输出结果映射过程相当于jdbc编程中对结果的解析处理过程。

7,Mybatis是如何将sql执行结果封装为目标对象并返回的?都有哪些映射形式?

第一种是使用<resultMap>标签,逐一定义列名和对象属性名之间的映射关系。

  

 

 

第二种是使用sql列的别名功能,将列别名书写为对象属性名,有了列名与属性名的映射关系后,Mybatis通过反射创建对象,同时使用反射给对象的属性逐一赋值并返回,那些找不到映射关系的属性,是无法完成赋值的。

(数据库中部门对象列为dname,实体为deptName  select  deptno,dname deptName,loc from dept )

 

8,Mybatis是否支持延迟加载?如果支持,它的实现原理是什么?

答:Mybatis仅支持association关联对象和collection关联集合对象的延迟加载,association指的就是一对一,多对一,collection指的就是一对多,多对多查询。在Mybatis配置文件中,可以配置是否启用延迟加载lazyLoadingEnabled=true|false。

    延迟加载代码实现(一对多collection 查询部门时,查出该部门的员工):

       实体:Dept   Emp

                  需要在Dept中加入private List<Emp> empList;  和getter setter

        daoDeptDao  

                 /**

     * 查询所有部门

     * @return

     */

    List<Dept>  listAll();

EmpDao:

   /**

     * 根据部门编号获取员工

     * @param deptNo

     * @return

     */

    List<Emp> listEmpByDeptNo(int deptNo);

 

        配置文件:DeptMapper.xml :

                  <!--查询所有部门-->

     <select id="listAll" resultMap="deptEmp">

          select  * from dept

     </select>

    <!--映射-->

    <resultMap id="deptEmp" type="com.aaa.mybatis.entity.Dept">

        <id property="deptNo" column="deptno"/>

        <result property="deptName" column="deptname"/>

        <result property="loc" column="loc"/>

        <collection property="empList" column="deptno" ofType="com.aaa.mybatis.entity.Emp"

        select="com.aaa.mybatis.dao.EmpDao.listEmpByDeptNo">

        </collection>

    </resultMap>

 EmpMapper.xml:

      <!--根据部门编号获取员工-->

    <select  id="listEmpByDeptNo" resultType="com.aaa.mybatis.entity.Emp">

        select empno,empname,salary from emp where deptno=#{deptNo}

    </select>

   主配置文件:mybatis-config.xml  中的 settings

             <!--开启全局懒加载-->

        <setting name="lazyLoadingEnabled" value="true"></setting>

         测试:

                 /**

     * 测试延时加载

     */

    @Test

    public void testListAll(){

        SqlSession sqlSession = null;

        try {

            sqlSession = SqlSessionFacotryUtil.getSqlSession();

            DeptDao deptDao = sqlSession.getMapper(DeptDao.class);

            List<Dept> depts = deptDao.listAll();

            if(depts!=null&&depts.size()>0){

                for (Dept dept : depts) {

                    dept.getEmpList();

                }

            }

        } catch (Exception e) {

            e.printStackTrace();

        } finally {

            if(sqlSession!=null)

                sqlSession.close();

        }

    }

如果下面代码注释掉,开启懒加载,底层查询数据库时,只执行了部门查询,如果不开启懒加载,部门和该部门的员工都查询了

   if(depts!=null&&depts.size()>0){

                for (Dept dept : depts) {

                    dept.getEmpList();

                }

            }

它的原理是,使用CGLIB创建目标对象的代理对象,当调用目标方法时,进入拦截器方法,比如调用a.getB().getName(),拦截器intercept()方法发现a.getB()是null值,那么就会单独发送事先保存好的查询关联B对象的sql,把B查询上来,然后调用a.setB(b),于是a的对象b属性就有值了,接着完成a.getB().getName()方法的调用。这就是延迟加载的基本原理。(ctrl+n   ProxyFactory    实现类:JavassistProxyFactory 的invoke和CglibProxyFactory的  intercept)

https://my.oschina.net/wenjinglian/blog/1857581?from=singlemessage

(Javaassist 就是一个用来 处理 Java 字节码的类库。)

当然了,不光是Mybatis,几乎所有的包括Hibernate,支持延迟加载的原理都是一样的。

9,模糊查询like语句该怎么写?

在Java代码中添加sql通配符。

 

      List<Dept> depts = deptDao.listByLike("%dev%");

       mapper中:

<!--模糊查询 like-->

    <select id="listByLike" parameterType="string" resultType="com.aaa.mybatis.entity.Dept">

        select  deptno,dname as deptName,loc from dept  where dname like  #{param}

    </select>

 在sql语句中拼接通配符

     select  deptno,dname as deptName,loc from dept  where dname like '%${param}%'

       '%#{param}%' 写法错误的

    select  deptno,dname as deptName,loc from dept  where dname like concat('%',# {param},'%')

   oracle   : '%'||# {param}||'%'    concat(concat('%',#{param}),'%')  

   mysql:  concat('%',# {param},'%')

   mysql 或者oracle  '%${param}%'

10,请问MyBatis中的动态SQL是什么意思?

对于一些复杂的查询,我们可能会指定多个查询条件,但是这些条件可能存在也可能不存在,需要根据用户指定的条件动态生成SQL语句。如果不使用持久层框架我们可能需要自己拼装SQL语句,还好MyBatis提供了动态SQL的功能来解决这个问题。MyBatis中用于实现动态SQL的元素主要有:

- if

- choose / when / otherwise

- trim

- where

- set

- foreach

 

11,请说明一下MyBatis中命名空间(namespace)的作用是什么?

在大型项目中,可能存在大量的SQL语句,这时候为每个SQL语句起一个唯一的标识(ID)就变得并不容易了。为了解决这个问题,在MyBatis中,可以为每个映射文件起一个唯一的命名空间,这样定义在这个映射文件中的每个SQL语句就成了定义在这个命名空间中的一个ID。只要我们能够保证每个命名空间中这个ID是唯一的,即使在不同映射文件中的语句ID相同,也不会再产生冲突了。

 12,如何执行批量插入?

    SQL 基础:

       -- 批量插入

--  mysql

insert into dept values (18,'dt1','1floor'),(19,'dt1','1floor'),(20,'dt1','1floor');

-- oracle

insert into dept

select 18,'dt1','1floor' from dual UNION

select 19,'dt1','1floor' from dual UNION

select 20,'dt1','1floor' from dual

dao:

/**

     * 批量插入

     * @param depts

     * @return

     */

    int  batchAdd(List<Dept> depts);

 mapper:

        <!--批量插入-->

    <insert id="batchAdd">

         insert into dept values

        <foreach collection="list" item="dept" index="i"   separator=",">

            (null,#{dept.deptName},#{dept.loc})

        </foreach>

    </insert>

测试:

       List<Dept> deptList = new ArrayList<Dept>(5);

            Dept dept = new Dept();

            dept.setDeptName("team33");

            dept.setLoc("33floor");

            deptList.add(dept);

            Dept dept1 = new Dept();

            dept1.setDeptName("team44");

            dept1.setLoc("44floor");

            deptList.add(dept1);

            Dept dept2 = new Dept();

            dept2.setDeptName("team55");

            dept2.setLoc("55floor");

            deptList.add(dept2);

            Dept dept3 = new Dept();

            dept3.setDeptName("team66");

            dept3.setLoc("66floor");

            deptList.add(dept3);

            Dept dept4 = new Dept();

            dept4.setDeptName("team77");

            dept4.setLoc("77floor");

            deptList.add(dept4);

            int i = deptDao.batchAdd(deptList);

            sqlSession.commit();

 

13,如何获取自动生成的(主)键值?

  MySQL:Mapper文件insert语句设置    useGeneratedKeys="true" keyProperty="id"

     <!--添加部门并返回自增ID-->

    <insert id="add" useGeneratedKeys="true" keyProperty="deptNo">

        insert into dept values(null,#{deptName},#{loc})

    </insert>

     自增生成之后的值赋值实体中的属性deptNo

Oracle:Mapper文件insert语句增加

<selectKey keyProperty="id" order="BEFORE" resultType="Integer">

    select xxx_SEQ.nextval from dual

</selectKey>

 14,为什么说Mybatis是半自动ORM映射工具?它与全自动的区别在哪里?

     hql:" from Emp" Emp 实体类

                session.save(emp);

               session.saveOrUpdate(emp);

              session.update(emp);

              session.deleteByke(empno);

               session.delete(emp); 

       Hibernate属于全自动ORM映射工具,使用Hibernate查询关联对象或者关联集合对象时,可以根据对象关系模型直接获取,所以它是全自动的(hibernate可以操作实体直接查询,不需要手写语句)。而Mybatis在查询关联对象或关联集合对象时,需要手动编写sql来完成,所以,称之为半自动ORM映射工具。

 

15,在mapper中如何传递多个参数?

(1)第一种:#{0}或者是#{arg0} 看是否是3.4.2及之前#{0} 或者之后#{arg0}

      dao:

          /**

     * 添加,传递多个参数

     * @param dname

     * @param loc

     * @return

     */

    int addA(String dname,String loc);

mapper:

       <!-- 添加,多个参数  3.4.2版本之前#{0}   之后#{arg0}    https://www.cnblogs.com/zhangmingcheng/p/9922236.html-->

    <insert id="addA">

        insert into dept values(null,#{arg0},#{arg1})

    </insert>

 

(2)第二种: 使用 @param 注解:

  dao:

       /**

     * 添加,@Param传递多个参数

     * @param dname

     * @param loc

     * @return

     */

    int addB(@Pa <!--添加,@Param传递多个参数-->

mapper:

    <insert id="addB">

        insert into dept values(null,#{deptName},#{loc})

    </insert>ram("deptName") String dname,@Param("loc") String loc);

 

 

 

 

 

(3)第三种:多个参数封装成map或者是实体

          dao方法:

   /**

     * 添加

     * @param dept

     * @return

     */

    int add(Dept dept);

    mapper:

   <!--添加部门并返回自增ID-->

    <insert id="add" useGeneratedKeys="true" keyProperty="deptNo">

        insert into dept values(null,#{deptName},#{loc})

    </insert>

 

  

15, Mybatis的Xml映射文件中,不同的Xml映射文件,id是否可以重复?

不同的Xml映射文件,如果配置了namespace,那么id可以重复;如果没有配置namespace,那么id不能重复;

 

原因就是namespace+id是作为Map<String, MapperStatement>的key使用的,如果没有namespace,就剩下id,那么,id重复会导致数据互相覆盖。有了namespace,自然id就可以重复,namespace不同,namespace+id自然也就不同。

 

但是,在以前的Mybatis版本的namespace是可选的,不过新版本的namespace已经是必须的了。

 一对一、一对多的关联查询 ? 

      参考以前上课时的例子(参考刚才讲的懒加载的例子也行)

 

16, Mybatis的一级、二级缓存

       参考项目(mybatis_interview_cache)

 

1)一级缓存: 基于 PerpetualCache 的 HashMap 本地缓存,其存储作用域为 Session,当 Session flush 或 close 之后,该 Session 中的所有 Cache 就将清空,默认打开一级缓存。

 

2)二级缓存与一级缓存其机制相同,默认也是采用 PerpetualCache,HashMap 存储,不同在于其存储作用域为 Mapper(Namespace),并且可自定义存储源,如 Ehcache。默认不打开二级缓存,要开启二级缓存,使用二级缓存属性类需要实现Serializable序列化接口(可用来保存对象的状态),可在它的映射文件中配置<cache/> ;

 

3)对于缓存数据更新机制,当某一个作用域(一级缓存 Session/二级缓存Namespaces)的进行了C/U/D 操作后,默认该作用域下所有 select 中的缓存将被 clear 掉并重新更新,如果开启了二级缓存,则只根据配置判断是否刷新

 17,使用MyBatis的mapper接口调用时有哪些要求?

 

①  Mapper接口方法名和mapper.xml中定义的每个sql的id相同;

②  Mapper接口方法的输入参数类型和mapper.xml中定义的每个sql 的parameterType的类型相同(配置文件中可以省略,如果写,必须相同);

③  Mapper接口方法的输出参数类型和mapper.xml中定义的每个sql的resultType的类型相同(通过resultMap也可以,更新方法没有返回值配置的);

④  Mapper.xml文件中的namespace即是mapper接口的类路径。

 

 

 

18, Xml 映射文件中,除了常见的 select|insert|updae|delete 标签之外,还有哪些标签?

还有很多其他的标签,<resultMap>、<parameterMap>、<sql>、<include>、<selectKey>,加上 动态 sql 的 9 个标签,trim|where|set|foreach|if|choose|when|otherwise|bind 等,其中<sql>为 sql 片段标签,通过<include>标签引入 sql 片段,<selectKey>为不支持自增的主键生成策略标签。

 

19,通常一个 Xml 映射文件,都会写一个 Dao 接口与之对应,请问,这个 Dao 接口的工作原理是什么?Dao接口里的方法,参数不同时,方法能重载吗?

Dao 接口,就是人们常说的 Mapper 接口,接口的全限名,就是映射文件中的 namespace 的值,接口的方法名,就是映射文件中 MappedStatement 的 id 值,接口方法内的参数,就是传递给 sql 的参数。Mapper 接口是没有 实 现 类 的 , 当 调 用 接 口 方 法 时 , 接 口 全 限 名 + 方 法 名 拼 接 字 符 串 作 为 key 值 , 可 唯 一 定 位 一 个MappedStatement,举例:com.mybatis3.mappers.StudentDao.findStudentById,可以唯一找到

namespace 为 com.mybatis3.mappers.StudentDao 下 面 id = findStudentById 的

MappedStatement。在 Mybatis 中,每一个<select>、<insert>、<update>、<delete>标签,都会被解

析为一个 MappedStatement 对象。

Dao 接口里的方法,是不能重载的,因为是全限名+方法名的保存和寻找策略。

 

Dao 接口的工作原理是 JDK 动态代理,Mybatis 运行时会使用 JDK 动态代理为 Dao 接口生成代理 proxy 对象,代理对象 proxy 会拦截接口方法,转而执行MappedStatement 所代表的 sql,然后将 sql 执行结果返回。

    DeptDao deptDao = sqlSession.getMapper(DeptDao.class);

https://www.cnblogs.com/hopeofthevillage/p/11384848.html

 

20 Mybatis都有哪些Executor执行器?它们之间的区别是什么?

答:Mybatis有三种基本的Executor执行器,SimpleExecutor、ReuseExecutor、BatchExecutor。

SimpleExecutor:每执行一次update或select,就开启一个Statement对象,用完立刻关闭Statement对象。

ReuseExecutor:执行update或select,以sql作为key查找Statement对象,存在就使用,不存在就创建,用完后,不关闭Statement对象,而是放置于Map<String, Statement>内,供下一次使用。简言之,就是重复使用Statement对象。

BatchExecutor:执行update(没有select,JDBC批处理不支持select),将所有sql都添加到批处理中(addBatch()),等待统一执行(executeBatch()),它缓存了多个Statement对象,每个Statement对象都是addBatch()完毕后,等待逐一执行executeBatch()批处理。与JDBC批处理相同。

作用范围:Executor的这些特点,都严格限制在SqlSession生命周期范围内。

21 Mybatis中如何指定使用哪一种Executor执行器?

     答:在Mybatis配置文件中,可以指定默认的ExecutorType执行器类型,也可以手动给DefaultSqlSessionFactory的创建SqlSession的方法传递ExecutorType类型参数。

         // sqlSessionFactory.openSession(ExecutorType.SIMPLE);

         //  sqlSessionFactory.openSession(ExecutorType.REUSE);

         // sqlSessionFactory.openSession(ExecutorType.BATCH);

 

 

 

22 Mybatis是如何进行分页的?分页插件的原理是什么?PagerHelper

 

        Mybatis使用RowBounds对象进行分页,它是针对ResultSet结果集执行的内存分页,而非物理分页。可以在sql内直接书写带有物理分页的参数来完成物理分页功能,也可以使用分页插件来完成物理分页。

 

       分页插件的基本原理是使用Mybatis提供的插件接口,实现自定义插件,在插件的拦截方法内拦截待执行的sql,然后重写sql,根据dialect方言,添加对应的物理分页语句和物理分页参数。

 

mybatis分页插件(非官网)

1,添加jar包

        <!-- 分页插件pagehelper -->        <dependency>            <groupId>com.github.pagehelper</groupId>            <artifactId>pagehelper</artifactId>            <version>5.0.0</version>        </dependency>        <dependency>            <groupId>com.github.pagehelper</groupId>            <artifactId>pagehelper-spring-boot-autoconfigure</artifactId>            <version>1.2.3</version>        </dependency>        <dependency>            <groupId>com.github.pagehelper</groupId>            <artifactId>pagehelper-spring-boot-starter</artifactId>            <version>1.2.3</version>        </dependency>      <!-- 分页插件pagehelper -->

2,添加springboot配置

 

              https://pagehelper.github.io/

https://github.com/pagehelper/Mybatis-PageHelper/blob/master/README_zh.md

 #分页插件

#helperDialect属性来指定分页插件使用哪种方言

pagehelper.helper-dialect=oracle

#当该参数设置为 true 时,pageNum<=0 时会查询第一页, pageNum>pages(超过总数时),会查询最后一页。

pagehelper.reasonable=true

#支持通过 Mapper 接口参数来传递分页参数,默认值false,分页插件会从查询方法的参数值中,自动根据上面 params 配置的字段中取值,查找到合适的值时就会自动分页。

pagehelper.support-methods-arguments=true

#增加了该参数来配置参数映射,用于从对象中根据属性名取值

pagehelper.params=count=countSql

3,具体使用

       与以前controller区别

  //设置当前第几页和每页显示数量    PageHelper.startPage(Integer.valueOf(map.get("pageNo")+""),Integer.valueOf(map.get("pageSize")+""));

        //用PageInfo对结果进行包装

        PageInfo<Map> pageInfo =new PageInfo<Map>(newsService.getList());

 

/**

     * 分页部门查询

     * @param map

     * @return

     */

    @ResponseBody

    @RequestMapping("page")

    public Object page(@RequestParam Map map){

        int pageNo = Integer.valueOf(map.get("pageNo")+"");

        int pageSize = Integer.valueOf(map.get("pageSize")+"");

        //初始化配置

        PageHelper.offsetPage(pageNo,pageSize);

        PageInfo<Map> pageInfo = new PageInfo<Map>(deptService.getList());

       //如果使用easyui可以这样封装,其他框架,自己根据pageInfo解析

        Map tmap  = new HashMap();

        tmap.put("total",pageInfo.getTotal());

        tmap.put("rows",pageInfo.getList());

        return tmap;

    }

posted @ 2020-08-19 09:40  漫漫夏夜  阅读(487)  评论(0编辑  收藏  举报