原文:

最近重构以前写的服务,最大的一个变动是将mybatis切换为spring data jpa,切换的原因很简单,有两点:第一、它是spring的子项目能够和spring boot很好的融合,没有xml文件(关于这一点hibernate似乎也很符合);第二、简单优雅,比如不需要写SQL、对分页有自动化的支持等等,基于以上两点开始了重构之路。在这之前了解了一下hibernate、mybatis和spring data jpa的区别,在这里也简单的记录一下:Hibernate的O/R Mapping实现了POJO 和数据库表之间的映射,以及SQL 的自动生成和执行;Mybatis则在于POJO 与SQL之间的映射关系,通过ResultMap对SQL的结果集进行映射;Spring Data jpa是一个用于简化数据库访问,并支持云服务的开源框架,容易上手,通过命名规范、注解查询简化查询操作。这三者都是ORM框架,但是mybatis可能并没有那么典型,原因就是mybatis映射的是SQL的结果集,另外hibernate和spring data jpa都是jpa(Java Persistence API,是从JDK5开始提供的,用来描述对象 <--> 关系表映射关系,并持久化的标准)的一种实现,从这一点上将这两者是一种并列的关系,spring data jpa用户手册上有这么一句话Improved compatibility with Hibernate 5.2.,这说明,spring data jpa又是hibernate的一个提升,下边先通过一个SQL:select * from User where name like '?' and age > ?的例子说明一下这三者在执行时候的区别:
首先看hibernate:

  1. public interface UserDao{
  2. List<User> findByNameLikeAndAgeGreaterThan(String firstName,Integer age);
  3. }
  4.  
  5. public class UserDaoImpl implements UserDao{
  6. @Override
  7. public List<User> findByFirstNameAndAge(String firstName, Integer age) {
  8. //具体hql查找:"from User where name like '%'"+firstName + "and age > " + age;
  9. return hibernateTemplateMysql.execute(new HibernateCallback() {
  10. @Override
  11. public Object doInHibernate(Session session) throws HibernateException {
  12. String hql = "from User where name like '?' and age > ?";
  13. Query query = session.createQuery(hql);
  14. query.setParameter(0, firstName+"");
  15. query.setParameter(1, age);
  16. return query.uniqueResult();
  17. }
  18. });
  19. }
  20. }

其次是mybatis:

  1. @Mapper
  2. public interface UserMapper {
  3. Increment findByNameLikeAndAgeGreaterThan(String name,int age);
  4. }
  5.  
  6. <select id="findByNameLikeAndAgeGreaterThan" parameterType="java.lang.Integer" resultMap="UserMap">
  7. select
  8. u.*
  9. from
  10. user u
  11. <where>
  12. u.name like ?1 and u.age>?2
  13. </where>
  14. </select>
  15.  
  16. <resultMap id="UserMap" type="com.txxs.po.User">
  17. <result column="id" property="id"/>
  18. <result column="name" property="name"/>
  19. <result column="age" property="age"/>
  20. </resultMap>

最后是spring data jpa:

  1. public interface UserDao extends JpaRepository<User, Serializable>{
  2. List<User> findByNameLikeAndAgeGreaterThan(String name,Integer age);
  3. }
  4. //为了增加代码的可读性可以使用注解的方式,这样方法的命名就不需要严格按照规范
  5. @Query("select * from User u where u.name like ?1 and u.age>?2")

通过上边代码的对比我们可以发现spring data jpa只要按照规范使用起来非常简单,下边是spring data jpa方法的命名规范:其他的可以参考用户手册
Keyword      Sample                                 JPQL snippet
And        findByLastnameAndFirstname                      … where x.lastname = ?1 and x.firstname = ?2

Or          findByLastnameOrFirstname                        … where x.lastname = ?1 or x.firstname = ?2

Is,Equals     findByFirstname,findByFirstnameIs,findByFirstnameEquals           … where x.firstname = ?1

Between      findByStartDateBetween                        … where x.startDate between ?1 and ?2

LessThan      findByAgeLessThan                          … where x.age < ?1

LessThanEqual    findByAgeLessThanEqual                        … where x.age <= ?1

GreaterThan    findByAgeGreaterThan                           … where x.age > ?1

GreaterThanEqual  findByAgeGreaterThanEqual                      … where x.age >= ?1

After         findByStartDateAfter                          … where x.startDate > ?1

Before        findByStartDateBefore                          … where x.startDate < ?1

IsNull        findByAgeIsNull                              … where x.age is null

IsNotNull,NotNull    findByAge(Is)NotNull                          … where x.age not null

Like          findByFirstnameLike                         … where x.firstname like ?1

NotLike        findByFirstnameNotLike                        … where x.firstname not like ?1

StartingWith      findByFirstnameStartingWith                      … where x.firstname like ?1(parameter bound with appended %)

EndingWith      findByFirstnameEndingWith                      … where x.firstname like ?1(parameter bound with prepended %)

Containing        findByFirstnameContaining                    … where x.firstname like ?1(parameter bound wrapped in %)

OrderBy        findByAgeOrderByLastnameDesc                    … where x.age = ?1 order by x.lastname desc

Not          findByLastnameNot                        … where x.lastname <> ?1

In            findByAgeIn(Collection<Age> ages)                  … where x.age in ?1

NotIn          findByAgeNotIn(Collection<Age> ages)                … where x.age not in ?1

True          findByActiveTrue()                          … where x.active = true

False          findByActiveFalse()                        … where x.active = false

IgnoreCase        findByFirstnameIgnoreCase                    … where UPPER(x.firstame) = UPPER(?1)

下边记录一下切换的服务的后台架构,分为四层:controller、service、repository以及mapper,这样在修改的时候只修改repository即可,并添加新的层dao层,这样只要通过repository的切换就可以快速的实现spring data jpa和mybatis的快速切换,甚至可以同时使用这两个框架,从框架层面解决了切换的问题之后,由于spring data jpa的更新和添加是相似的两个方法,所以把所有的添加、批量添加、更新和批量更新抽象为以下的两个方法:

  1. @Repository
  2. public class CommonRepository<T> {
  3.  
  4. @PersistenceContext
  5. protected EntityManager entityManager;
  6.  
  7. /**
  8. * 添加和批量添加数据
  9. * @param lists
  10. */
  11. @Transactional
  12. public void batchAddCommon(List<T> lists){
  13. int size = lists.size();
  14. for (int i = 0; i < size; i++) {
  15. entityManager.persist(lists.get(i));
  16. if (i % 100 == 0 || i == (size - 1)) {
  17. entityManager.flush();
  18. entityManager.clear();
  19. }
  20. }
  21. }
  22.  
  23. /**
  24. * 数据的批量更新
  25. * @param lists
  26. */
  27. @Transactional
  28. public void batchUpdate(List<T> lists) {
  29. int size = lists.size();
  30. for (int i = 0; i < size; i++) {
  31. entityManager.merge(lists.get(i));
  32. if (i % 100 == 0 || i == (size - 1)) {
  33. entityManager.flush();
  34. entityManager.clear();
  35. }
  36. }
  37. }
  38. }

  

从这一点上讲spring data jpa会比mybatis要强很多,因为以上两个方法可以实现所有资源的更新和添加操作,而mybatis则需要为每一个资源实体去写添加、批量添加、更新和批量更新等,这会很麻烦。以下是切换过程中一些有记录意义的SQL,罗列一下:

  1. //修改方法和删除方法都需要添加@Modifying,占位符是从1开始而不是开始的
  2. @Modifying
  3. @Query("update table n set n.column1 =?1 where n.column2 = ?2")
  4. Integer updateObject(String one,String two);
  5.  
  6. @Modifying
  7. @Query("delete from table n where n.column1 = ?1")
  8. Integer getObject(String one);
  9.  
  10. //查询某一个字段的时候需要制定相应的类型,select全量数据的使用直接使用别名n即可,原生的SQL需要使用n.*
  11. @Query("select n.column1 as String from table n where n.column2 is null or n.column2 =''")
  12. List<String> getObject();
  13.  
  14. //原生SQL,进行了连表操作,并且查询了满足数组条件
  15. @Query(value="select s.*, i.* from table1 s, table2 i where i.column1 = s.column1 and i.column1 in (?1) order by s.id desc", nativeQuery = true)
  16. List<Server> getObject(List<Integer> arry);

在切换的使用遇到一个比较复杂的SQL,涉及联表、查询参数变量、in、case when、分页、group by等,下边给出mybatis和spring data jpa的写法:

  1. <select id="queryNotUsedObject" parameterType="com.txxs.po.Object" resultType="java.lang.Integer" >
  2. select
  3. DISTINCT (i.column1),
  4. SUM(CASE WHEN i.column7=#{column7} THEN 1 ELSE 0 END) used,
  5. sum(CASE WHEN i.column7 IS NULL THEN 1 ELSE 0 END) not_used
  6. from
  7. table1 i,
  8. table2 s
  9. <where>
  10. <if test="column2 != null and column2 != '' ">
  11. and s.column2 = #{column2}
  12. </if>
  13. <if test="column3 != null and column3 != '' ">
  14. and s.column3 = #{column3}
  15. </if>
  16. <if test="column4 != null and column4 != '' ">
  17. and i.column4 like '${column4}%'
  18. </if>
  19. <if test="column5 != null and column5 != '' ">
  20. and i.column5 like '${column5}%'
  21. </if>
  22. <if test="column6 != null and column6 != '' ">
  23. and i.column6 like '${column6}%'
  24. </if>
  25. and s.column1 = i.column1
  26. </where>
  27. GROUP BY column1
  28. having used =0 and not_used>0
  29. ORDER BY s.id DESC
  30. <if test="page != null and page>=0" >
  31. limit #{page} , #{size}
  32. </if>
  33. </select>

spring data jpa方式:

  1. public Page<Object> queryNotUsedObject(final Object query){
  2. CriteriaBuilder criteriaBuilder = entityManager.getCriteriaBuilder();
  3. CriteriaQuery criteriaQuery = criteriaBuilder.createQuery();
  4. //查询的根
  5. Root<Server> root = criteriaQuery.from(entityManager.getMetamodel().entity(Object.class));
  6. //判断参数
  7. List<Predicate> predicates = new ArrayList<Predicate>();
  8. if(null != query.getColumn1()){
  9. predicates.add(criteriaBuilder.equal(root.get("Column1"), query.getColumn1()));
  10. }
  11. if(null != query.getColumn2()){
  12. predicates.add(criteriaBuilder.equal(root.get("Column2"), query.getColumn2()));
  13. }
  14. //联表操作
  15. if(null != query.getColumn3()){
  16. predicates.add(criteriaBuilder.equal(root.join("table1Column").get("Column3"), query.getColumn3()));
  17. }
  18. if(null != query.getColumn4()){
  19. predicates.add(criteriaBuilder.equal(root.join("table1Column").get("Column4"), query.getColumn4()));
  20. }
  21. if(null != query.getColumn5()){
  22. predicates.add(criteriaBuilder.equal(root.join("table1Column").get("Column5"), query.getColumn5()));
  23. }
  24. //拼接Sum
  25. Expression<Integer> sumExpOne = criteriaBuilder.sum(criteriaBuilder.<Integer>selectCase().when(criteriaBuilder.equal(root.join("table1Column").get("Column6"), query.getColumn6()), 1).otherwise(0)).as(Integer.class);
  26. Expression<Integer> sumExpTwo = criteriaBuilder.sum(criteriaBuilder.<Integer>selectCase().when(criteriaBuilder.isNull(root.join("table1Column").get("Column6")), 1).otherwise(0)).as(Integer.class);
  27. //查询参数
  28. List<Expression<?>> expressions = new ArrayList<Expression<?>>();
  29. expressions.add(root.join("table1Column").get("Column1"));
  30. //having参数
  31. List<Predicate> predicateArrayList = new ArrayList<Predicate>();
  32. Predicate predicate = criteriaBuilder.equal(sumExpOne,0);
  33. predicate = criteriaBuilder.and(predicate,criteriaBuilder.greaterThan(sumExpTwo,0));
  34. predicateArrayList.add(predicate);
  35. //拼接SQL
  36. criteriaQuery.multiselect(expressions.toArray(new Expression[expressions.size()])).distinct(true);
  37. criteriaQuery.where(predicates.toArray(new Predicate[predicates.size()]));
  38. criteriaQuery.groupBy(root.join("table1Column").get("Column1"));
  39. criteriaQuery.having(predicateArrayList.toArray(new Predicate[predicateArrayList.size()]));
  40. //获取第一次执行的结果
  41. final List<Integer> list = entityManager.createQuery(criteriaQuery).getResultList();
  42.  
  43. Sort sort = new Sort(Sort.Direction.DESC, "id");
  44. Pageable objectDao.findAll(new Specification<Object>(){
  45. @Override
  46. public Predicate toPredicate(Root<Object> root, CriteriaQuery<?> criteriaQuery, CriteriaBuilder criteriaBuilder) {
  47. //判断参数
  48. List<Predicate> predicates = new ArrayList<Predicate>();
  49. predicates.add(root.get("id").in(list));
  50. return criteriaBuilder.and(predicates.toArray(new Predicate[predicates.size()]));
  51. }
  52. },pageable);
  53. }

上边代码里边很多column不对应,是为了隐去痕迹,方法已经测试通过,从上边的代码看spring data jpa对于复杂语句的支持不够,需要通过代码的方式实现,而这种方式代码的可读性会比较差,优化等都会有一些难度

最后总结一下就是如果业务简单实用spring data jpa即可,如果业务复杂还是实用mybatis吧

spring data jpa还是只使用简单的操作.

感觉最终还是要用mybatis,觉得jpa这种东西只适合简单的增删改查比较多、SQL不怎么变化的情况。

我们可以二者结合使用!!

实例对比 hibernate, spring data jpa, mybatis 选型参考的更多相关文章

  1. Spring data Jpa,Mybatis,读写锁,@Lock 使用

    Spring data jpa 支持注解式的读写锁(悲观锁),实际上这个东西硬编码也简单,但是基于Jpa 命名方式定义的Sql,只能用注解添加支持读写锁了, 不了解读写锁的可以点这里 mysql读写锁 ...

  2. 【hibernate spring data jpa】执行了save()方法 sql语句也执行了,但是数据并未插入数据库中

    执行了save()方法  sql语句也执行了,但是数据并未插入数据库中 解决方法: 是因为执行了save()方法,也执行了sql语句,但是因为使用的是 @Transactional 注解,不是手动去提 ...

  3. ORM框架 Mybatis、Hibernate、Spring Data JPA之到底该用谁,谁更牛*

    在持久层框架中无可厚非的就是mybatis了,但是也会经常被人问到为啥要用mybatis,为啥不用hibernate,jpa.很多人各级都是地铁爷爷看手机的表情,似乎从来没想过这个问题.“公司叫用我就 ...

  4. 来说说JPA、Hibernate、Spring Data JPA之间的什么关系?

    目录 JPA Hibernate Spring Data JPA 实践 来说说JPA.Hibernate.Spring Data JPA之间的什么关系 Java 持久层框架访问数据库的方式大致分为两种 ...

  5. Spring Boot 2.x基础教程:使用Spring Data JPA访问MySQL

    在数据访问这章的第一篇文章<Spring中使用JdbcTemplate访问数据库> 中,我们已经介绍了如何使用Spring Boot中最基本的jdbc模块来实现关系型数据库的数据读写操作. ...

  6. Hibernate、Mybatis与Spring Data JPA

    从零开始集成Springboot+MyBatis+JPA https://www.jianshu.com/p/e14c4a6f6871 MyBatis 与Hibernate的区别 http://xhr ...

  7. jdbc、jpa、spring data jpa、hibernate、mybatis之间的关系及区别

    基础概念 jdbc(Java DataBase Connectivity)是java连接数据库操作的原生接口.JDBC对Java程序员而言是API,对实现与数据库连接的服务提供商而言是接口模型.作为A ...

  8. Spring Data JPA、MyBatis还有Hibernate有什么区别

    原文:https://www.imooc.com/article/19754?block_id=tuijian_wz Spring Data JPA.MyBatis还有Hibernate有什么区别 2 ...

  9. MyBatis 与 Spring Data JPA 选择谁?

    MyBatis 与 Spring Data JPA 选择谁? https://www.v2ex.com/t/285081 jpa predicate优缺点 https://blog.csdn.net/ ...

随机推荐

  1. Vmware 安装centos7与网络配置

    一.下载linux镜像 下载地址:http://isoredirect.centos.org/centos/7/isos/x86_64/CentOS-7-x86_64-DVD-1804.iso 二.创 ...

  2. python之路-day1-if...else...流程判断

    判断输入的用户名:#Author:zww _username = "zww" _password = " username = input("username: ...

  3. opencv学习之路(1)、示例程序

    一.介绍 工欲善其事必先利其器,首先当然是配置环境安装软件啦.  我安装的vs2012+opencv2.48以及opencv3.0.具体安装步骤按照浅墨大神的博客进行即可:http://blog.cs ...

  4. ListView与ArrayAdapter(二)

    ArrayAdapter: 数组适配器,用于简单的文字列表 activity_main.xml <RelativeLayout xmlns:android="http://schema ...

  5. Python3基础 dict in/not in 查询一个字符是否指定字典的键或者值

             Python : 3.7.0          OS : Ubuntu 18.04.1 LTS         IDE : PyCharm 2018.2.4       Conda ...

  6. img的基线对齐问题

    http://blog.csdn.net/u011997156/article/details/44806523

  7. 赞 ( 84 ) 微信好友 新浪微博 QQ空间 180 SSD故事会(14):怕TLC因为你不了解!【转】

    本文转载自:https://diy.pconline.com.cn/750/7501340.html [PConline 杂谈]从前,大家谈TLC色变:如今,TLC攻占SSD半壁江山.是的,这个世界就 ...

  8. POJ 2400 Supervisor, Supervisee(KM二分图最大权值匹配)题解

    题意:n个老板n个员工,先给你n*n的数据,i行j列代表第i个老板第j喜欢的员工是谁,再给你n*n的数据,i行j列代表第i个员工第j喜欢的老板是谁,如果匹配到第k喜欢的人就会产生一个分数k-1.现在让 ...

  9. UVA11270 Tiling Dominoes(轮廓线动态规划)

    轮廓线动态规划是一种基于状态压缩解决和连通性相关的问题的动态规划方法 这道题是轮廓线动态规划的模板 讲解可以看lrj的蓝书 代码 #include <cstdio> #include &l ...

  10. P2475 [SCOI2008]斜堆(递归模拟)

    思路 可并堆真是一种神奇的东西 不得不说这道题是道好题,虽然并不需要可并堆,但是能加深对可并堆的理解 首先考虑斜堆的性质,斜堆和左偏树相似,有如下的性质 一个节点如果有右子树,就一定有左子树 最后插入 ...