Mybatis面试问题
一、JDBC编程有哪些不足的地方,Mybatis是如何解决这些问题的?
-
数据库连接的创建和销毁会造成系统资源浪费从而影响性能,如何使用数据库连接池来解决这个问题。
**解决:**在SqlMapConfig.xml中配置数据库连接池,使用连接池管理数据库连接。
-
Sql语句在编写时写在代码中造成代码不宜维护,实际使用的sql语句变化可能比较大,sql改动需要重新改写java代码。
解决:将Sql语句配置在XXXXMapper.xml文件中,实现于java代码分离。
-
向sql语句中传入参数较为困难,因为sql语句的where不一定,可能多可能少,占位符要和参数一一对应。
解决:Mybatis使用java对象映射到sql语句中。
-
对结果集解析麻烦,sql变化导致解析代码变化,且解析前需要遍历,如果能将数据库记录封装成pojo对象解析比较方便。
解决:Mybatis自动将sql执行结果映射至java对象。
二、Mybatis编程步骤是什么
- 创建SqlSessionFactory
- 通过SqlSessionFactory创建SqlSession
- 通过SqlSession执行数据库操作
- 调用session.commit()提交事务
- 调用session.close()关闭事务
三、Mybatis与Hibernate有什么不同
Mybatis并不是完全的ORM框架,因为Mybatis需要程序员自己编写对应的Sql语句,不过mybatis可以通过xml或者注解的当时灵活的配置要运行的sql语句,并将java对象和sql语句进行映射生成最终的sql语句,对sql语句返回的结果最终生成对应的java对象。
Mybatis学习门槛低,简单易学,直接编写原生的sql,可严格控制sql 的性能,灵活度高,非常适合对关系模型数据要求不高的软件开发,例如互联网软件、企业运行类软件等,这些软件与需求无关,一旦需求变化可以迅速更改数据库。但是灵活性的前提是mybatis无法做到数据库无关性,如果需要支持多种数据库的软件则需要自定义多套sql映射文件,工作量大。
Hibernate对象/关系映射能力强,数据库无关性好,对于关系模型要求高的软件(例如需求固定的定制化软件)如果使用hibernate开发可以节省很多代码,提高效率。但是Hibernate缺点是学习门槛高,精通门槛高,而且怎么O/R映射,在性能和对象模型之间平衡,以及怎么用好Hibernate都需要很高的经验和能力。
四、使用MyBatis的mapper接口调用时有哪些要求?
- Mapper接口方法名和mapper.xml中定义的每个sql的id相同
- Mapper接口方法的输入参数类型和mapper.xml中定义的每个sql的parameterType类型相同
- Mapper接口方法中输出的参数类型和mapper.xml中定义的每个sql的resultType相同
- Mapper.xml文件中的namespace即是Mapper接口的路径
五、SqlMapConfig.xml中配置有哪些内容?
SqlMapConfig.xml中配置的内容和顺序如下:
- properties(属性)
- setting(配置)
- typeAliases(类型别名)
- typeHandlers(类型处理器)
- objectFactory(对象工厂)
- plugins(插件)
- environments(环境集合属性对象)
- environment(环境子属性对象)
- transactionManager(事务管理)
- dataSource(数据源)
- mapper(映射器)
六、简单的说一下MyBatis的一二级缓存?
MyBatis首先去缓存中查询结果集,如果没有则查询数据库,如果有从缓存中取出数据集,MyBatis内部缓存存储使用HashMap,key存放在CacheKey
中使用其中的hashcode
private static final int DEFAULT_HASHCODE = 17;
默认hashCode为17
多个对象时,对每个对象计算HashCode,在进行混淆
int baseHashCode = object == null ? 1 : ArrayUtil.hashCode(object);
count++;
checksum += baseHashCode;
baseHashCode *= count;
hashcode = multiplier * hashcode + baseHashCode;
updateList.add(object);
SqlSession:该类是非线程安全的,其实例是不能共享的,所以应该每个线程具有自己的SqlSession实例。因此最佳建议是“请求或方法范围”。例如:收到一个Http请求后,创建一个实例,进行某些操作,之后关闭。确保将关闭放于finally中。
Mybatis的二级缓存即查询缓存,它的作用域是一个mapper的namespace,即在同一个namespace中查询sql可以从缓存中获取数据。二级缓存是可以跨SqlSession的,通过<cache>
开启二级缓存,通过<cache-ref>
关联两个不同的命名空间,使用二级缓存属性类需要实现Serializable序列化接 口(可用来保存对象的状态)。
七、#{}
和${}
有什么区别
-
#{}解析传递过来的参数数据
-
${}
对传递进来的参数拼接到SQL中 -
#{}
是预编译的,${}是字符串替换。 -
使用#{}可以有效防止SQL注入,提高系统安全性
-
需要动态拼接表名时。
SELECT * FROM ${tablename}
-
动态拼接排序字段。
SELECT * FROM TABLE ORDER BY ${username}
八、如何获取自动生成的(主)键值?
需求:user对象在插入数据库后,新记录的主键要通过user对象返回,通过user获取主键值。
解决思路:
通过LAST_INSERT_ID()获取刚插入的自增主键值,在insert语句执行后,执行select LAST_INSERT_ID()就可以获取自增主键。
Mysql
<insert id="insertUser" parameterType="cn.itcast.mybatis.po.User">
<selectKey keyProperty="id" order="AFTER" resultType="int">
select LAST_INSERT_ID()
</selectKey>
INSERT INTO USER(username,birthday,sex,address) VALUES(#{username},#{birthday},#{sex},#{address})
</insert>
Oracle:
实现思路:
先查询序列获得主键,将主键设置到user对象中,将user对象插入到数据库中。
<insert id="insertUser" parameterType="cn.itcast.mybatis.po.User">
<selectKey keyProperty="id" order="BEFORE" resultType="int">
select 序列.nextval() from dual
</selectKey>
INSERT INTO USER(id,username,birthday,sex,address) VALUES( 序列.nextval(),#{username},#{birthday},#{sex},#{address})
</insert>
Mybatis动态sql是做什么的?
-
Mybatis动态sql可以让我们在xml映射文件内,以标签的形式编写动态的sql,完成逻辑判断和动态拼接sql 的功能。
-
Mybatis提供9中动态sql标签:
trim|where|set|foreach|if|choose|when|otherwise|bind。
-
执行原理,使用OGNL从sql参数对象中计算表达式的值,根据表达式的值动态拼接sql,以此来完成动态sql的功能。
九、Mybatis的方法重载
Mybatis同一个mapper下面定义的相同的方法名,不同入参的函数,是不能进行重载的。因为使用的HashMap的Key为packageName+methodName。重名时会出现错误。
十、Mybatis基本结构
十一、参考
作者:沉沦2014
链接:https://www.jianshu.com/p/b7c591494fe8
來源:简书
简书著作权归作者所有,任何形式的转载都请联系作者获得授权并注明出处。