MyBatis 常见面试题37道-包含答案_mybatis面试,2024年最新架构师必备

先自我介绍一下,小编浙江大学毕业,去过华为、字节跳动等大厂,目前阿里P7

深知大多数程序员,想要提升技能,往往是自己摸索成长,但自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年最新Golang全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友。
img
img
img
img
img

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

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

如果你需要这些资料,可以添加V获取:vip1024b (备注go)
img

正文

  • #{} 的变量替换是在DBMS 中;${} 的变量替换是在 DBMS 外
18. 模糊查询like语句该怎么写

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

select id,sex,age,username,password from person where username LIKE{pattern}
19. 在mapper中如何传递多个参数

方法1:顺序传参法

public User selectUser(String name, int deptId);

select \* from user where user_name ={

0} and dept_id ={

1}

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

方法2:@Param注解传参法

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

select \* from user where user_name ={

userName} and dept_id ={

deptId}

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

方法3:Map传参法

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

select \* from user where user_name ={

userName} and dept_id ={

deptId}

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

方法4:Java Bean传参法

public User selectUser(User user);

select \* from user where user_name ={

userName} and dept_id ={

deptId}

  • #{}里面的名称对应的是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
  • 具体用法如下:
//推荐使用 INSERT INTO emp(ename,gender,email,did) VALUES (#{emp.eName},#{emp.gender},#{emp.email},#{emp.dept.id}) INSERT INTO emp(ename,gender,email,did) VALUES(#{emp.eName},#{emp.gender},#{emp.email},#{emp.dept.id})
  • 使用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 into employee(lastName,email,gender) values(#{lastName},#{email},#{gender})
21. 如何获取生成的主键
  • 新增标签中添加:keyProperty=" ID " 即可
insert into user(user_name, user_password, create_time) values(#{userName},{userPassword} ,{createTime, jdbcType=TIMESTAMP})

 

22. 当实体类中的属性名和表中的字段名不一样 ,怎么办
  • 第1种: 通过在查询的SQL语句中定义字段名的别名,让字段名的别名和实体类的属性名一致。
select order_id id, order_no orderno ,order_price price form orders where order_id=#{id};
  • 第2种: 通过 来映射字段名和实体类属性名的一一对应的关系。
select \* from orders where order_id=#{

id}


<!–用id属性来映射主键字段–>

<!–用result属性来映射非主键字段,property为实体类属性名,column为数据库表中的属性–>


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

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

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

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

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

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

  • 第三种:使用 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扫描器:;

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能执行一对多,一对一的联系查询吗,有哪些实现方法
  • 能,不止可以一对多,一对一还可以多对多,一对多
  • 实现方式:1、 单独发送一个SQL去查询关联对象,赋给主对象,然后返回主对象;2、 使用嵌套查询,似JOIN查询,一部分是A对象的属性值,另一部分是关联对象B的属性值,;好处是只要发送一个属性值,就可以把主对象和关联对象查出来3、 子查询;
33. Mybatis是否可以映射Enum枚举类?

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

需要这份系统化的资料的朋友,可以添加V获取:vip1024b (备注Go)
img

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

  • 实现方式:1、 单独发送一个SQL去查询关联对象,赋给主对象,然后返回主对象;2、 使用嵌套查询,似JOIN查询,一部分是A对象的属性值,另一部分是关联对象B的属性值,;好处是只要发送一个属性值,就可以把主对象和关联对象查出来3、 子查询;
33. Mybatis是否可以映射Enum枚举类?

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

需要这份系统化的资料的朋友,可以添加V获取:vip1024b (备注Go)
[外链图片转存中…(img-TOcogNIv-1713145688276)]

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

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值