优点:
缺点:
Hibernate 属于全自动ORM映射工具,使用 Hibernate 查询关联对象或者关联集合对象时,可以根据对象关系模型直接获取,所以它是全自动的。
Mybatis 在查询关联对象或关联集合对象时,需要手动编写 SQL 来完成,所以,称之为半自动ORM映射工具。
相同点:都是对 Jdbc 的封装,都是持久层的框架,都用于 Dao 层的开发。
不同点
映射关系
MyBatis 是一个半自动映射的框架,配置 Java 对象与 SQL 语句执行结果的对应关系,多表关联关系配置简单。
Hibernate 是一个全表映射的框架,配置 Java 对象与数据库表的对应关系,多表关联关系配置复杂。
SQL优化和移植性
Hibernate 对 SQL 语句封装,提供了日志、缓存、级联(级联比 MyBatis 强大)等特性,此外还提供 HQL(Hibernate Query Language)操作数据库,数据库无关性支持好,但会多消耗性能。如果项目需要支持多种数据库,代码开发量少,但SQL语句优化困难。
MyBatis 需要手动编写 SQL,支持动态 SQL、处理列表、动态生成表名、支持存储过程。开发工作量相对大些。直接使用SQL语句操作数据库,不支持数据库无关性,但 SQL 语句优化容易。
开发难易程度和学习成本
Hibernate 是重量级框架,学习使用门槛高,适合于需求相对稳定,中小型的项目,比如:办公自动化系统。MyBatis 是轻量级框架,学习使用门槛低,适合于需求变化频繁,大型的项目,比如:互联网电子商务系统。
总结:MyBatis 是一个小巧、方便、高效、简单、直接、半自动化的持久层框架,Hibernate 是一个强大、方便、高效、复杂、间接、全自动化的持久层框架。
1、数据库链接创建、释放频繁造成系统资源浪费从而影响系统性能,如果使用数据库链接池可解决此问题。
解决:在SqlMapConfig.xml中配置数据链接池,使用连接池管理数据库链接。
2、SQL 语句写在代码中造成代码不易维护,实际应用 SQL 变化的可能较大,SQL 变动需要改变 Java代码。
解决:将 SQL 语句配置在XXXXmapper.xml文件中与 Java 代码分离。
3、 向 SQL 语句传参数麻烦,因为 SQL 语句的where条件不一定,可能多也可能少,占位符需要和参数一一对应。
解决: Mybatis自动将 Java对象映射至 SQL 语句。
4、 对结果集解析麻烦,SQL 变化导致解析代码变化,且解析前需要遍历,如果能将数据库记录封装成 POJO 对象解析比较方便。
解决:Mybatis自动将 SQL 执行结果映射至 Java 对象。
【MyBatis占位符】Mybatis中的#{}和${}区别
Dao 接口即 Mapper 接口。接口的全限名就是映射文件中的 namespace 的值;
接口的方法名,就是映射文件中 Mapper 的 Statement 的 id 值;
接口方法内的参数,就是传递给 SQL 的参数。
Mapper 接口是没有实现类的,当调用接口方法时,接口全限名+方法名的拼接字符串作为 key 值,可唯一定位一个MapperStatement。
Dao接口里的方法,是不能重载的,因为是全限名+方法名的保存和寻找策略。
Dao接口的工作原理是 JDK动态代理,Mybatis 运行时会使用 JDK 动态代理为 Dao 接口生成代理 proxy 对象,代理对象 proxy 会拦截接口方法,转而执行 MappedStatement 所代表的 SQL ,然后将 SQL 执行结果返回。
若Dao 层函数有多个参数,那么其对应的XML中,#{0}代表接收的是 Dao 层中的第一个参数,#{1}代表 Dao 中的第二个参数,以此类推。
使用@Param注解:在Dao层的参数中前加@Param注解,注解内的参数名为传递到Mapper中的参数名。
多个参数封装成Map,以 HashMap 的形式传递到 Mapper 中。
Mybatis动态SQL可以在XML映射文件内,以标签的形式编写动态SQL,执行原理是根据表达式的值完成逻辑判断,并动态拼接SQL的功能。
Mybatis提供了9种动态SQL标签:trim、where、set、foreach、if、choose、when、otherwise、bind
其执行原理为,使用 OGNL(Object Graph Navigation Language) 从 SQL 参数对象中计算表达式的值,根据表达式的值动态拼接sql,以此来完成动态sql的功能。
不同的XML映射文件,如果配置了namespace,那么 id 可以重复;如果没有配置 namespace,那么 id 不能重复;
原因是 namespace+id 是作为 Map
的 key 使用的,如果没有 namespace,就剩下 id,那么 id 重复会导致数据互相覆盖。有了 namespace,自然 id 就可以重复,namespace 不同,namespace+id 自然也不同。
有联合查询和嵌套查询两种方式。
联合查询是几个表联合查询,通过在 resultMap 里面配置 association 节点配置一对一的类就可以完成;
嵌套查询是先查一个表,根据这个表里面的结果的外键 id,再去另外一个表里面查询数据,也是通过association 配置,但另外一个表的查询是通过 select 配置的。
有联合查询和嵌套查询两种方式。
联合查询是几个表联合查询,只查询一次,通过在 resultMap 里面的 collection 节点配置一对多的类就可以完成;
嵌套查询是先查一个表,根据这个表里面的结果的外键 id,再去另外一个表里面查询数据,也是通过collection,但另外一个表的查询是通过 select 配置的。
对于缓存数据更新机制,当某一个作用域(一级缓存 Session/二级缓存 Namespace)进行了增/删/改操作后,默认该作用域下所有 select 中的缓存将被 clear。