91,什么是ORM?
对象关系映射( Object-Relational-Mapping,简称ORM)是一种为解决程序面向对象模型和数据库的关系模型不匹配问题的一种技术;
简单的说,ORM是通过使用描述对象和数据库枝江映射的元数据(在Java中可以用XML或者注解),将程序中的对象自动持久化到关系数据库中或者将关系数据库表中的行转化成Java对象,其本质就是将数据从一种形式转换到另外一种形式。
92,Hibernate中SessionFactory是线程安全的吗?Session是线程安全的吗(两个线程能够共享同一个Session吗)?
SessionFactory对应Hibernate的一个数据存储的概念,它是线程安全的,可以被多个线程并发访问。SessionFactory一般只会在启动的时候构建。对于应用程序最好将SessionFactory通过单例模式进行封装以便于访问。
Session是一个轻量级非线程安全的对象(线程间不能共享session),它表示与数据库进行交互的一个工作单元。Session是由SessionFactory创建的,在创建任务完成之后它会被关闭。Session是持久层对外提供的主要接口。
Session会延迟获取数据库连接(也就是在需要的时候才会获取)。为了避免创建太多的session,可以使用ThreadLocal将session和当前线程绑定在一起,这样可以让同一个线程获得的总是同一个session。Hibernate3 中SessionFactory的getCurrentSession()方法就可以做到。
94,阐述Session加载实体对象的过程。
1、Session在调用数据库查询功能之前,首先会在一级缓存中通过实体类和主键查找,如果一级缓存命中且数据状态合法,则直接返回;
2、如果一级缓存没有命中,接下来Session会在当前NonExists记录(相当于一个查询黑名单,如果出现重复的无效查询可以迅速做出判断,从而提升性能)中进行查询,如果NonExists中存在同样的查询条件,则返回null;
3、如果一级缓存查询失败查询二级缓存,如果二级缓存命中直接返回;
4、如果之前的查询都未命中,则发出SQL语句,如果查询未发现对应记录则将此次查询添加到Session的NonExists中加以记录,返回null;
5、根据映射配置和SQL语句得到ResultSet,并创建对应的实体对象;
6、将对象纳入Session(一级缓存)的管理;
7、如果有对应的拦截器,则执行拦截器的onLoad方法;
8、如果开启并设置了要使用二级缓存,则将对象纳入二级缓存;
9、返回数据对象。
95,MyBatis中使用#和$书写占位符有什么区别?
#将传入的数据当成一个字符串,会对传入的数据自动加上引号;
$将传入的数据直接显示生成在SQL中。
注意:使用$占位符可能会导致SQL注入攻击,能用#的地方就不要用$,写order by字句的时候应该用$而不是#。
96,解释一下MyBatis中命名空间(namespace)的作用。
在大型项目中,可能存在大量的SQL语句,这时候为每个SQL语句起一个唯一的标识(ID)将变得不容易了。为了解决这个问题,在MyBatis中,可以为每个映射文件起一个唯一的命名空间,这样定义在这个映射文件中的每个SQL语句就成了定义在这个命名空间的一个ID。只要我们能够保证每个命名空间中的这个ID是唯一的,即便在不同的映射文件中的语句ID相同,也不会再产生冲突了。
97、MyBatis中的动态SQL是什么意思?
对于一些复杂的查询,我们可能会指定多个查询条件,但是这些条件可能存在也可能不存在,如果不使用持久层框架我们可能需要自己拼装SQL语句,不过MyBatis提供了动态SQL的功能来解决这个问题。MyBatis中用于实现动态SQL的元素主要有:
- if - choose / when / otherwise - trim - where - set - foreach
用法举例:
<select id="foo" parameterType="Blog" resultType="Blog">
select * from t_blog where 1 = 1
<if test="title != null">
and title = #{title}
</if>
<if test="content != null">
and content = #{content}
</if>
<if test="owner != null">
and owner = #{owner}
</if>
</select>
98,JDBC编程有哪些不足之处,MyBatis是如何解决这些问题的?
1、JDBC:数据库连接创建、释放频繁造成系统资源浪费从而影响系统性能,如果使用数据库连接池可以解决此问题。
MyBatis:在SqlMapConfig.xml中配置数据库连接池,使用连接池管理数据库连接。
2、JDBC:Sql语句写在代码中造成代码不易维护,实际应用sql变化的可能较大,sql变动需要改变java代码。
MyBatis:将Sql语句配置在XXXXmapper.xml文件中与java代码分离。
3、JDBC:向sql语句传参数麻烦,因为sql语句的where条件不一定,可能多也可能少,占位符需要和参数一一对应。
MyBatis:Mybatis自动将java对象映射至sql语句。
4、JDBC:对结果集解析麻烦,sql变化导致解析代码变化,且解析前需要遍历,如果能将数据库记录封装成pojo对象解析比较方便。
MyBatis:Mybatis自动将sql执行结果集映射至java对象。
99,MyBatis与Hibernate有哪些不同?
1、MyBatis和Hibernate不同,它不完全是一个ORM框架,应为MyBatis需要程序员自己编写Sql语句,不过mybatis可以通过XML或者注解方式灵活配置要运行的sql语句,并将java对象和sql语句映射生成最终的sql,最后将sql执行的结果再映射生成java对象。
2、MyBatis学习门槛低,简单易学,程序员直接写原生态sql,可以严格控制sql执行性能,灵活度高,非常适合对关系数据模型要求不高的软件开发,例如互联网软件,企业运营软件等,应为这类软件需求变化频繁,一旦需求变化要求成果输出迅速。但是灵活的前提是MyBatis无法做到数据库无关性,如果需要实现支持多种数据库的软件则需要自定义多套sql映射文件,工作量大。
3、Hibernate对象关系映射能力强,数据库无关性好,对于关系模型要求高的软件(例如需求固定的定制化软件)如果用Hibernate开发可以节省很多代码,提高效率。但是Hibernate的缺点是学习门槛高,要精通门槛更高,而且怎么设计O/R映射,在性能和对象模型之间如何权衡,以及怎样用好Hibernate需要具有很强的经验和能力才行。
总之,按照用户的需求在有限的资源环境下只要能做出维护性、扩展性良好的软件架构都是好架构,所以框架适合才是最好。
(这里也可以结合自己的理解说,别说的收不住)
100,简单的说一下MyBatis的一级缓存和二级缓存?
MyBatis首先去缓存中查询结果集,如果没有则查询数据库,如果有则从缓存取出返回结果集就不走数据库。MyBatis内部存储缓存使用一个HashMap,Key为hashCode+sqlId+Sql语句。value为从查询出来映射生成的java对象。
MyBatis的二级缓存即为查询缓存,它的作用域是一个mapper的namespace,即在同一个namespace中查询sql可以从缓存中获取数据。二级缓存可以跨SQLSession的。