2024年最全MyBatis 常见面试题37道-包含答案_mybatis面试,Golang研发岗必问30+道高级面试题

img
img

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化的资料的朋友,可以添加戳这里获取

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

17. #{}和${}的区别
  • #{}是占位符,预编译处理;${}是拼接符,字符串替换,没有预编译处理。
  • Mybatis在处理#{}时,#{}传入参数是以字符串传入,会将SQL中的#{}替换为?号,调用PreparedStatement的set方法来赋值。
  • #{} 可以有效的防止SQL注入,提高系统安全性;${} 不能防止SQL 注入
  • #{} 的变量替换是在DBMS 中;${} 的变量替换是在 DBMS 外
18. 模糊查询like语句该怎么写

1’%${question}%’ 可能引起SQL注入,不推荐2 “%”#{question}“%” 注意:因为#{…}解析成sql语句时候,会在变量外侧自动加单引号’ ',所以这里% 需要使用双引号" ",不能使用单引号 ’ ',不然会查不到任何结果。3 CONCAT(’%’,#{question},’%’) 使用CONCAT()函数,(推荐)4使用bind标签(不推荐)

<select id="listUserLikeUsername" resultType="com.jourwon.pojo.User">
	<bind name="pattern" value="'%' + username + '%'" />
	select id,sex,age,username,password from person where username LIKE{pattern}
</select>


19. 在mapper中如何传递多个参数

方法1:顺序传参法

public User selectUser(String name, int deptId);

<select id="selectUser" resultMap="UserResultMap">
    select \* from user where user_name ={

0} and dept_id ={

1}
</select>


  • #{}里面的数字代表传入参数的顺序。
  • 这种方法不建议使用,sql层表达不直观,且一旦顺序调整容易出错。

方法2:@Param注解传参法

public User selectUser(@Param("userName") String name, int @Param("deptId") deptId);

<select id="selectUser" resultMap="UserResultMap">
    select \* from user where user_name ={

userName} and dept_id ={

deptId}
</select>


  • #{}里面的名称对应的是注解@Param括号里面修饰的名称。
  • 这种方法在参数不多的情况还是比较直观的,(推荐使用)。

方法3:Map传参法

public User selectUser(Map<String, Object> params);

<select id="selectUser" parameterType="java.util.Map" resultMap="UserResultMap">
    select \* from user where user_name ={

userName} and dept_id ={

deptId}
</select>


  • #{}里面的名称对应的是Map里面的key名称。
  • 这种方法适合传递多个参数,且参数易变能灵活传递的情况。(推荐使用)。

方法4:Java Bean传参法

public User selectUser(User user);

<select id="selectUser" parameterType="com.jourwon.pojo.User" resultMap="UserResultMap">
    select \* from user where user_name ={

userName} and dept_id ={

deptId}
</select>


  • #{}里面的名称对应的是User类里面的成员属性。
  • 这种方法直观,需要建一个实体类,扩展不容易,需要加属性,但代码可读性强,业务逻辑处理方便,推荐使用。(推荐使用)。
20. Mybatis如何执行批量操作
  • 使用foreach标签
  • foreach的主要用在构建in条件中,它可以在SQL语句中进行迭代一个集合。foreach标签的属性主要有item,index,collection,open,separator,close。
  • item 表示集合中每一个元素进行迭代时的别名,随便起的变量名;
  • index 指定一个名字,用于表示在迭代过程中,每次迭代到的位置,不常用;
  • open 表示该语句以什么开始,常用“(”;
  • separator 表示在每次进行迭代之间以什么符号作为分隔符,常用“,”;
  • close 表示以什么结束,常用“)”。
  • 在使用foreach的时候最关键的也是最容易出错的就是collection属性,该属性是必须指定的,但是在不同情况下,该属性的值是不一样的,主要有一下3种情况:1、 如果传入的是单参数且参数类型是一个List的时候,collection属性值为list;2、 如果传入的是单参数且参数类型是一个array数组的时候,collection的属性值为array;3、 如果传入的参数是多个的时候,我们就需要把它们封装成一个Map了,当然单参数也可以封;装成map,实际上如果你在传入参数的时候,在MyBatis里面也是会把它封装成一个Map的,map的key就是参数名,所以这个时候collection属性值就是传入的List或array对象在自己封装的map里面的key
  • 具体用法如下:
<!-- 批量保存(foreach插入多条数据两种方法)
int addEmpsBatch(@Param("emps") List<Employee> emps); -->
<!-- MySQL下批量保存,可以foreach遍历 mysql支持values(),(),()语法 --> //推荐使用
<insert id="addEmpsBatch">
    INSERT INTO emp(ename,gender,email,did)
    VALUES
    <foreach collection="emps" item="emp" separator=",">
        (#{emp.eName},#{emp.gender},#{emp.email},#{emp.dept.id})
    </foreach>
</insert>


<!-- 这种方式需要数据库连接属性allowMutiQueries=true的支持
如jdbc.url=jdbc:mysql://localhost:3306/mybatis?allowMultiQueries=true -->
<insert id="addEmpsBatch">
    <foreach collection="emps" item="emp" separator=";">
        INSERT INTO emp(ename,gender,email,did)
            VALUES(#{emp.eName},#{emp.gender},#{emp.email},#{emp.dept.id})
    </foreach>
</insert>


  • 使用ExecutorType.BATCH
  • Mybatis内置的ExecutorType有3种,默认为simple,该模式下它为每个语句的执行创建一个新的预处理语句,单条提交sql;而batch模式重复使用已经预处理的语句,并且批量执行所有更新语句,显然batch性能将更优; 但batch模式也有自己的问题,比如在Insert操作时,在事务没有提交之前,是没有办法获取到自增的id,这在某型情形下是不符合业务要求的
  • 具体用法如下:
//批量保存方法测试
@Test
public void testBatch() throws IOException{

    SqlSessionFactory sqlSessionFactory = getSqlSessionFactory();
    //可以执行批量操作的sqlSession
    SqlSession openSession = sqlSessionFactory.openSession(ExecutorType.BATCH);
    //批量保存执行前时间
    long start = System.currentTimeMillis();
    try {

        EmployeeMapper mapper = openSession.getMapper(EmployeeMapper.class);
        for (int i = 0; i < 1000; i++) {

            mapper.addEmp(new
            Employee(UUID.randomUUID().toString().substring(0, 5), "b", "1"));
        }
        openSession.commit();
        long end = System.currentTimeMillis();
        //批量保存执行后的时间
        System.out.println("执行时长" + (end - start));
        //批量 预编译sql一次==》设置参数==》10000次==》执行1次 677
        //非批量 (预编译=设置参数=执行 )==》10000次 1121
    } finally {

        openSession.close();
    }
}


  • mapper和mapper.xml如下
public interface EmployeeMapper {

    //批量保存员工
    Long addEmp(Employee employee);
}


<mapper namespace="com.jourwon.mapper.EmployeeMapper"
    <!--批量保存员工 -->
    <insert id="addEmp">
        insert into employee(lastName,email,gender)
            values(#{lastName},#{email},#{gender})
    </insert>
</mapper>


21. 如何获取生成的主键
  • 新增标签中添加:keyProperty=" ID " 即可
<insert id="insert" useGeneratedKeys="true" keyProperty="userId" >
    insert into user(user_name, user_password, create_time)
    values(#{userName},{userPassword} ,{createTime, jdbcType=TIMESTAMP})
</insert>


 

22. 当实体类中的属性名和表中的字段名不一样 ,怎么办
  • 第1种: 通过在查询的SQL语句中定义字段名的别名,让字段名的别名和实体类的属性名一致。
<select id="getOrder" parameterType="int" resultType="com.jourwon.pojo.Order">
    select order_id id, order_no orderno ,order_price price form orders
    where order_id=#{id};
</select>


  • 第2种: 通过 来映射字段名和实体类属性名的一一对应的关系。
<select id="getOrder" parameterType="int" resultMap="orderResultMap">
    select \* from orders where order_id=#{

id}
</select>
<resultMap type="com.jourwon.pojo.Order" id="orderResultMap">
    <!–用id属性来映射主键字段–>
    <id property="id" column="order\_id">
    <!–用result属性来映射非主键字段,property为实体类属性名,column为数据库表中的属性–>
    <result property ="orderno" column ="order\_no"/>
    <result property="price" column="order\_price" />
</reslutMap>


23. Mapper 编写有哪几种方式?
  • 第一种:接口实现类继承 SqlSessionDaoSupport:使用此种方法需要编写mapper 接口,mapper 接口实现类、mapper.xml 文件。

1、 在sqlMapConfig.xml中配置mapper.xml的位置;

<mappers>
    <mapper resource="mapper.xml 文件的地址" />
    <mapper resource="mapper.xml 文件的地址" />
</mappers>


1、 定义mapper接口;2、 实现类集成SqlSessionDaoSupport;mapper 方法中可以 this.getSqlSession()进行数据增删改查。
3、 spring配置;

<bean id=" " class="mapper 接口的实现">
    <property name="sqlSessionFactory"ref="sqlSessionFactory"></property>
</bean>


  • 第二种:使用 org.mybatis.spring.mapper.MapperFactoryBean:

1、 在sqlMapConfig.xml中配置mapper.xml的位置,如果mapper.xml和mappre接口的名;称相同且在同一个目录,这里可以不用配置
2、 定义mapper接口:;

<mappers>
    <mapper resource="mapper.xml 文件的地址" />
    <mapper resource="mapper.xml 文件的地址" />
</mappers>


1、 mapper.xml中的namespace为mapper接口的地址;2、 mapper接口中的方法名和mapper.xml中的定义的statement的id保持一致;3、 Spring中定义;

<bean id="" class="org.mybatis.spring.mapper.MapperFactoryBean">
    <property name="mapperInterface" value="mapper 接口地址" />
    <property name="sqlSessionFactory" ref="sqlSessionFactory" />
</bean>


  • 第三种:使用 mapper 扫描器:

1、 mapper.xml文件编写:;mapper.xml 中的 namespace 为 mapper 接口的地址;mapper 接口中的方法名和 mapper.xml 中的定义的 statement 的 id 保持一致;如果将 mapper.xml 和 mapper 接口的名称保持一致则不用在 sqlMapConfig.xml中进行配置。
2、 定义mapper接口:;注意 mapper.xml 的文件名和 mapper 的接口名称保持一致,且放在同一个目录
3、 配置mapper扫描器:;

<bean class="org.mybatis.spring.mapper.MapperScannerConfigurer">
	<property name="basePackage" value="mapper 接口包地址"></property>
	<property name="sqlSessionFactoryBeanName" value="sqlSessionFactory"/>
</bean>


1、 使用扫描器后从spring容器中获取mapper的实现对象;

24. 什么是MyBatis的接口绑定?有哪些实现方式?
  • 接口绑定,就是在MyBatis中任意定义接口,然后把接口里面的方法和SQL语句绑定,我们直接调用接口方法就可以,这样比起原来了SqlSession提供的方法我们可以有更加灵活的选择和设置。
  • 接口绑定有两种实现方式1、 通过注解绑定,就是在接口的方法上面加上@Select、@Update等注解,里面包含Sql语句来;绑定;2、 通过xml里面写SQL来绑定,在这种情况下,要指定xml映射文件里面的namespace必须为;接口的全路径名。当Sql语句比较简单时候,用注解绑定, 当SQL语句比较复杂时候,用xml绑定,一般用xml绑定的比较多。
25. 使用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接口的类路径;

26. 这个Dao接口的工作原理是什么?Dao接口里的方法,参数不同时,方法能重载吗?
  • Dao接口的工作原理是JDK动态代理,Mybatis运行时会使用JDK动态代理为Dao接口生成代理proxy对象,代理对象proxy会拦截接口方法,转而执行MappedStatement所代表的sql,然后将sql执行结果返回。
  • Dao接口里的方法,是不能重载的,因为是全限名+方法名的保存和寻找策略。
27. Mybatis的Xml映射文件中,不同的Xml映射文件,id是否可以重复?
  • 不同的Xml映射文件,如果配置了namespace,那么id可以重复;如果没有配置namespace,那么id不能重复;毕竟namespace不是必须的,只是最佳实践而已。
  • 原因就是namespace+id是作为Map<String, MappedStatement>的key使用的,如果没有namespace,就剩下id,那么,id重复会导致数据互相覆盖。有了namespace,自然id就可以重复,namespace不同,namespace+id自然也就不同。
28. 简述Mybatis的Xml映射文件和Mybatis内部数据结构之间的映射关系?
  • 答:Mybatis将所有Xml配置信息都封装到All-In-One重量级对象Configuration内部。在Xml映射文件中, 标签会被解析为ParameterMap对象,其每个子元素会被解析为ParameterMapping对象。 标签会被解析为ResultMap对象,其每个子元素会被解析为ResultMapping对象。每一个 、 、 、 标签均会被解析为MappedStatement对象,标签内的sql会被解析为BoundSql对象。
29. Mybatis是如何将sql执行结果封装为目标对象并返回的?都有哪些映射形式?
  • 第一种是使用 标签,逐一定义列名和对象属性名之间的映射关系。
  • 第二种是使用sql列的别名功能,将列别名书写为对象属性名,比如T_NAME AS NAME,对象属性名一般是name,小写,但是列名不区分大小写,Mybatis会忽略列名大小写,智能找到与之对应对象属性名,你甚至可以写成T_NAME AS NaMe,Mybatis一样可以正常工作。

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

30. Xml映射文件中,除了常见的select|insert|updae|delete标签之外,还有哪些标签?
  • 还有很多其他的标签, <resultMap><parameterMap><sql><include><selectKey> ,加上动态sql的9个标签,trim|where|set|foreach|if|choose|when|otherwise|bind等,其中 为sql片段标签,通过<include>标签引入sql片段,<selectKey> 为不支持自增的主键生成策略标签。
31. Mybatis映射文件中,如果A标签通过include引用了B标签的内容,请问,B标签能否定义在A标签的后面,还是说必须定义在A标签的前面?
  • 虽然Mybatis解析Xml映射文件是按照顺序解析的,但是,被引用的B标签依然可以定义在任何地方,Mybatis都可以正确识别。
  • 原理是,Mybatis解析A标签,发现A标签引用了B标签,但是B标签尚未解析到,尚不存在,此时,Mybatis会将A标签标记为未解析状态,然后继续解析余下的标签,包含B标签,待所有标签解析完毕,Mybatis会重新解析那些被标记为未解析的标签,此时再解析A标签时,B标签已经存在,A标签也就可以正常解析完成了。
32. Mybatis能执行一对多,一对一的联系查询吗,有哪些实现方法

img
img
img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上Go语言开发知识点,真正体系化!

由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新

如果你需要这些资料,可以戳这里获取

Mybatis会重新解析那些被标记为未解析的标签,此时再解析A标签时,B标签已经存在,A标签也就可以正常解析完成了。

32. Mybatis能执行一对多,一对一的联系查询吗,有哪些实现方法

[外链图片转存中…(img-IPAOf272-1715718067517)]
[外链图片转存中…(img-IJZ91v0k-1715718067518)]
[外链图片转存中…(img-iWHUyDtM-1715718067519)]

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上Go语言开发知识点,真正体系化!

由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新

如果你需要这些资料,可以戳这里获取

  • 20
    点赞
  • 19
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Mybatis-Plus 是一个基于 Mybatis 的增强工具,它简化了 Mybatis 的开发流程,提供了许多实用的功能,如自动生成代码、分页查询、条件构造器、性能分析等。在 2021 Mybatis-Plus 面试中,可能会涉及到以下问题: 1. Mybatis-Plus 的优点是什么? Mybatis-Plus 的优点包括:简化开发流程、提高开发效率、提供实用的功能、易于集成、支持多种数据库、性能优秀等。 2. Mybatis-Plus 的核心功能是什么? Mybatis-Plus 的核心功能包括:自动生成代码、分页查询、条件构造器、性能分析、多租户支持、逻辑删除等。 3. Mybatis-Plus 的代码生成器是什么?有什么作用? Mybatis-Plus 的代码生成器是一个可视化工具,可以根据数据库表结构自动生成实体类、Mapper 接口、Mapper XML 文件等代码,大大提高了开发效率。 4. Mybatis-Plus 的分页查询是如何实现的? Mybatis-Plus 的分页查询是通过 PageHelper 类实现的,它可以自动拦截 SQL 语句,将查询结果封装成 Page 对象,提供了丰富的分页查询方法。 5. Mybatis-Plus 的条件构造器是什么?有什么作用? Mybatis-Plus 的条件构造器是一个灵活的查询条件构造工具,可以根据不同的查询需求,动态生成 SQL 语句,支持多种查询条件,如等于、不等于、大于、小于、模糊查询等。 6. Mybatis-Plus 的性能分析是如何实现的? Mybatis-Plus 的性能分析是通过 PerformanceInterceptor 类实现的,它可以拦截 SQL 语句,统计 SQL 执行时间、执行次数、执行的 SQL 语句等信息,帮助开发者优化 SQL 语句的性能。 7. Mybatis-Plus 的多租户支持是什么?有什么作用? Mybatis-Plus 的多租户支持是指可以根据不同的租户,动态切换数据源,实现数据隔离的功能。它可以帮助开发者在多租户场景下,简化数据访问的流程,提高开发效率。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值