1. 程式人生 > >SpringBoot中使用Spring Data Jpa 實現簡單的動態查詢的兩種方法

SpringBoot中使用Spring Data Jpa 實現簡單的動態查詢的兩種方法

ppr eat value table 得到 blog .net ride integer

首先謝謝大佬的簡書文章: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//@[email protected]
    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

SpringBoot中使用Spring Data Jpa 實現簡單的動態查詢的兩種方法