• SpringBoot中使用Spring Data Jpa 实现简单的动态查询的两种方法


      软件152 尹以操

    首先谢谢大佬的简书文章:http://www.jianshu.com/p/45ad65690e33#

    这篇文章中讲的是spring中使用spring data jpa,使用了xml配置文件。我现在使用的是spring boot ,没有了xml文件配置就方便多了。我同样尝试了两种方式,也都是简单的查询,需要更复杂的查询,还需要我研究研究。往下看,需要先配置springboot的开发环境,需要大致了解springboot,这里可以看下面两篇文章:

    springboot 项目新建

    springboot使用小记

    创建实体类:

    package com.entity;
    
    import javax.persistence.Entity;
    import javax.persistence.GeneratedValue;
    import javax.persistence.Id;
    import javax.persistence.Table;
    
    /**
     *create by yyc 2017年6月11日上午9:59:59
     */
    @Entity
    @Table(name="test_user")
    public class TestUser {
    
        @Id
        @GeneratedValue
        private int userId;
        private Integer userAge;
        private String  userName;
        private Integer high;////省略getter、setter
        
    }

    为了测试,先创建一个简单的实体类。

    写元数据模型:

    Criteria API

    这套API可用于构建对数据库的查询。

    类型安全。通过定义元数据模型,在程序编译阶段就可以对类型进行检查,不像SQL需要与Mysql进行交互后才能发现类型问题。

    如下即为元数据模型。创建一个元模型类,类名最后一个字符为下划线,内部的成员变量与UserInfo.class这个实体类的属性值相对应。

    package com.entity;
    
    import javax.persistence.metamodel.SingularAttribute;
    import javax.persistence.metamodel.StaticMetamodel;
    import com.TestUser;
    @StaticMetamodel(TestUser.class)
    public class TestUser_ {
    
         public static volatile SingularAttribute<TestUser, Integer> userId;// 用户ID,自增量
         public static volatile SingularAttribute<TestUser, Integer> userAge;
         public static volatile SingularAttribute<TestUser, String> userName;
         public static volatile SingularAttribute<TestUser, Integer> high;
    
        }

    可移植。API并不依赖具体的数据库,可以根据数据库类型的不同生成对应数据库类型的SQL,所以其为可移植的。
    面向对象。Criteria API是使用的是各种类和对象如CriteriaQuery、Predicate等构建查询,是面向对象的。而如果直接书写SQL则相对于面向的是字符串。

    现在开始SpringBoot中使用Spring Data Jpa 实现简单的动态查询的两种方式方法

    第一种方式:通过JPA的Criteria API实现

    1. EntityManager获取CriteriaBuilder
    2. CriteriaBuilder创建CriteriaQuery
    3. CriteriaQuery指定要查询的表,得到Root<UserInfo>,Root代表要查询的表
    4. CriteriaBuilder创建条件Predicate,Predicate相对于SQL的where条件,多个Predicate可以进行与、或操作。
    5. 通过EntityManager创建TypedQuery
    6. TypedQuery执行查询,返回结果
    package com.repository;
    
    import java.util.List;
    import javax.persistence.EntityManager;
    import javax.persistence.PersistenceContext;
    import javax.persistence.criteria.CriteriaBuilder;
    import javax.persistence.criteria.CriteriaQuery;
    import javax.persistence.criteria.Predicate;
    import javax.persistence.criteria.Root;
    import org.springframework.beans.factory.annotation.Autowired;
    import org.springframework.stereotype.Repository;
    import com.entity.TestUser;
    import com.entity.TestUser_;
    
    /**
     *create by yyc 2017年6月11日下午9:08:39
     */
    @Repository
    public class TestUserExtendDao {
    
    //    @Autowired
        @PersistenceContext//@Autowired和@PersistenceContext注解任取一
        EntityManager em;
        
        @SuppressWarnings("unused")
        public List<TestUser> getTestUserInfo(final Integer age, final String name, final Integer high){
             //1
            CriteriaBuilder cb = em.getCriteriaBuilder();
             //2
            CriteriaQuery<TestUser> query = cb.createQuery(TestUser.class);
            
            //3
            //from 
            Root<TestUser> root = query.from(TestUser.class);
            
             //4
             //where
            Predicate p1 = null;
            if (age != 0) {
                System.out.println("正在操作age!!!");
                Predicate p2 = cb.equal(root.get(TestUser_.userAge), age);
                if (p1 != null) {
                    p1=cb.and(p1, p2);
                } else {
                    p1 = p2;
                }
            }
            if (false==name.isEmpty()) {
                System.out.println("正在操作name!!!");
                Predicate p2 = cb.equal(root.get(TestUser_.userName), name);
                if (p1 != null) {
                    p1=cb.and(p1, p2);
                } else {
                    p1 = p2;
                }
            }
            if (high != 0) {
                System.out.println("正在操作high!!!");
                Predicate p2 = cb.equal(root.get(TestUser_.high), high);
                if (p1 != null) {
                    p1=cb.and(p1, p2);
                } else {
                    p1 = p2;
                }
            }
            //5
            query.where(p1);
            //6
            List<TestUser> testUsers = em.createQuery(query).getResultList();
            return testUsers;
        }
    }
                                                                                            

    第二种方式:DAO层接口实现JpaSpecificationExecutor<T>接口

      JpaSpecificationExecutor如下,方法参数Specification接口有一个方法toPredicate,返回值正好是Criteria API中的Predicate,而Predicate相对于SQL的where条件。与上一个方法相比,这种写法不需要指定查询的表是哪一张,也不需要自己通过Criteria API实现排序和分页,只需要通过新建Pageable、Sort对象并传参给findAll方法即可,简便一些。

    这是JpaSpecificationExecutor接口中的方法:

    public interface JpaSpecificationExecutor<T> {
       T findOne(Specification<T> spec);
       List<T> findAll(Specification<T> spec);
       Page<T> findAll(Specification<T> spec, Pageable pageable);
       List<T> findAll(Specification<T> spec, Sort sort);
       long count(Specification<T> spec);
    }

    TestRepository继承JpaSpecificationExecutor接口:

    package com.repository;
    
    import org.springframework.data.jpa.repository.JpaSpecificationExecutor;
    import org.springframework.data.repository.PagingAndSortingRepository;
    import com.entity.TestUser;
    
    /**
     *create by yyc 2017年6月11日上午9:36:27
     *测试动态sql
     */
    public interface TestRepository extends PagingAndSortingRepository<TestUser, Integer>, JpaSpecificationExecutor<TestUser>{
    
    }

    实现Specification:

    package com.entity;
    
    import javax.persistence.criteria.CriteriaBuilder;
    import javax.persistence.criteria.CriteriaQuery;
    import javax.persistence.criteria.Predicate;
    import javax.persistence.criteria.Root;
    import org.springframework.data.jpa.domain.Specification;
    import com.entity.TestUser;
    
    /**
     * create by yyc 2017年6月11日上午10:17:44
     */
    public class TestUserDaoSpec {
        public static Specification<TestUser> getSpec(final Integer age, final String name, final Integer high) {
            return new Specification<TestUser>() {
    
                @SuppressWarnings("unused")
                @Override
                public Predicate toPredicate(Root<TestUser> root, CriteriaQuery<?> query, CriteriaBuilder cb) {
                    Predicate p1 = null;
                    if (age != 0) {
                        System.out.println("正在操作age!!!");
                        Predicate p2 = cb.equal(root.get(TestUser_.userAge), age);
                        if (p1 != null) {
                            p1=cb.and(p1, p2);
                        } else {
                            p1 = p2;
                        }
                    }
                    if (false==name.isEmpty()) {
                        System.out.println("正在操作name!!!");
                        Predicate p2 = cb.equal(root.get(TestUser_.userName), name);
                        if (p1 != null) {
                            p1=cb.and(p1, p2);
                        } else {
                            p1 = p2;
                        }
                    }
                    if (high != 0) {
                        System.out.println("正在操作high!!!");
                        Predicate p2 = cb.equal(root.get(TestUser_.high), high);
                        if (p1 != null) {
                            p1=cb.and(p1, p2);
                        } else {
                            p1 = p2;
                        }
                    }
                    return p1;
                }
            };
        }
    
    }

    Service层的调用测试类:

    package com.service.impl;
    
    import java.util.List;
    
    import org.springframework.beans.factory.annotation.Autowired;
    import org.springframework.stereotype.Service;
    import com.entity.TestUser;
    import com.entity.TestUserDaoSpec;
    import com.repository.TestRepository;
    import com.repository.TestUserExtendDao;
    import com.result.Result;//一个方法返回的封装,这里直接忽略即可
    
    
    /**
     *create by yyc 2017年6月11日上午9:40:51
     */
    @Service("testService")
    public class TestService {
    
        @Autowired
        private TestRepository testRepository;
        
        @Autowired
        private TestUserExtendDao testUserExtendDao;
        
        //测试第一种方式
        public Result getTestUsersByExtendDao(){
            List<TestUser> list = testUserExtendDao.getTestUserInfo(20, "", 170);//通过两个条件,string设为空
            printTestUserInfo(list);
            return new Result("查询成功!", list);
        }
        
        //测试第二种方式
        public Result getTestUsersByThreeParameter(){
            List<TestUser> list = testRepository.findAll(TestUserDaoSpec.getSpec(20, "yyc", 170));//通过三个条件
            printTestUserInfo(list);
            return new Result("查询成功!",list);
        }
        public Result getTestUsersByTwoParameter1(){
            List<TestUser> list = testRepository.findAll(TestUserDaoSpec.getSpec(20, "yyc", 0));//通过两个条件,Integer设为0
            printTestUserInfo(list);
            return new Result("查询成功!",list);
        }
    
        public Result getTestUsersByOneParameter(){
            List<TestUser> list = testRepository.findAll(TestUserDaoSpec.getSpec(0, "lrs", 0));//通过一个条件查询
            printTestUserInfo(list);
            return new Result("查询成功!",list);
        }
    
        private void printTestUserInfo(List<TestUser> list) {
            if (list!=null) {
                for (TestUser testUser : list) {
                    System.out.println("userId:"+testUser.getUserId()+
                            " userName:"+testUser.getUserName()+
                            " userAge:"+testUser.getUserAge()+
                            " userHigh:"+testUser.getHigh());
                }
            }
            
        }
    }

    再次感谢大佬们的文章:

    http://www.jianshu.com/p/45ad65690e33#

    http://blog.csdn.net/ie8848520/article/details/8161986

    http://www.cnblogs.com/jiangxiaoyaoblog/p/5635152.html

  • 相关阅读:
    JSDOM优化
    Firebug Console 与命令行全集
    input输入框和 pure框架中的 box-sizing 值问题
    模块化网站注意事项
    COOKIE
    鼠标滚动
    拖拽的基本函数(已有限制范围和修复浏览器默认行为以及磁性吸附、碰撞检测、改变层大小、模拟滚动条)
    app_can 的AJax异步,两个解决方案
    基于jQuery的message插件实现右下角弹出消息框
    C#后台讲字符串转化为计算公式
  • 原文地址:https://www.cnblogs.com/hyyq/p/6986797.html
Copyright © 2020-2023  润新知