亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

JPA如何使用nativequery多表關聯(lián)查詢返回自定義實體類

 更新時間:2021年11月18日 11:56:47   作者:lw5885799  
這篇文章主要介紹了JPA如何使用nativequery多表關聯(lián)查詢返回自定義實體類,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教

JPA nativequery多表關聯(lián)查詢返回自定義實體類

JPA官方推薦的多表關聯(lián)查詢使用不便,接觸的有些項目可能會使用JPA 做簡單查詢,Mybaits做復雜查詢。所以想要尋找一種好用的解決方案。

JPA多表關聯(lián)的實現(xiàn)方式

1.使用Specification實現(xiàn)映射關系匹配,如@ManyToOne等

2.使用NativeQuery等sql或hql來實現(xiàn)

優(yōu)缺點對比

1.映射關系是hibernate的入門基礎,很多人都會習慣去使用。個人不太喜歡這種方式,復用性太弱,且不靈活特別是在多表復雜業(yè)務情況下。

2.使用Specification方式需要繼承JpaSpecificationExecutor接口,構造對應的方法后傳入封裝查詢條件的Specification對象。邏輯上簡單易懂,但是構造Specification對象需要拼接格式條件非常繁瑣。

3.直接使用NativeQuery等方式實現(xiàn)復雜查詢個人比較喜歡,直觀且便利,弊端在于無法返回自定義實體類。需要手動封裝工具類來實現(xiàn)Object到目標對象的反射。

使用sql并返回自定義實體類

個人比較喜歡的實現(xiàn)方式,不多說看代碼

import org.springframework.stereotype.Repository;
import javax.persistence.EntityManager;
import javax.persistence.PersistenceContext;
import javax.transaction.Transactional;
 
@Repository
public class EntityManagerDAO {
    @PersistenceContext
    private EntityManager entityManager;
     /**
     * 人員列表排序
     * @return
     */
    @Transactional
    public List<BackstageUserListDTO> listUser(){
        String sql = "select a.create_time createTime," +
                "a.mobilephone phoneNum," +
                "a.email email,a.uid uid," +
                "a.enabled enabled," +
                "c.id_number idNumber," +
                " (case b.`status` when 1 then 1 else 0 end) status " +
                "from tbl_sys_user a " +
                "LEFT JOIN user_high_qic b on a.uid=b.u_id" +
                "LEFT JOIN user_qic c on a.uid=c.uid " +
                "ORDER BY status desc";
 
        SQLQuery sqlQuery = entityManager.createNativeQuery(sql).unwrap(SQLQuery.class);
        Query query = 
     sqlQuery.setResultTransformer(Transformers.aliasToBean(BackstageUserListDTO.class));
        List<BackstageUserListDTO> list = query.list();
        entityManager.clear();
        return list;
    }
}
public class BackstageUserListDTO implements Serializable{
    private static final long serid = 1L;
    private String createTime;
    private String phoneNum;
    private String email;
    private BigInteger uid;
    private Integer enabled;
    private String idNumber;
    private BigInteger status;
    //GETTER SETTER
}

這樣一個需求如果使用前兩種方式實現(xiàn),無疑會非常麻煩。使用這種方式能夠直接反射需要的自定義實體類。

可以根據(jù)需求整理封裝成不同的方法,加入排序,分頁等。我在這里主要提供一種方便的解決思路。

JPA多表關聯(lián)動態(tài)查詢(自定義sql語句)

項目需求,查詢需求數(shù)據(jù)需要多表鏈接——>根據(jù)多種條件篩選查詢到的數(shù)據(jù),在網(wǎng)上查了很多資料最終選擇這個字符串拼接查詢

類似如此動態(tài)查詢

[外鏈圖片轉存失敗,源站可能有防盜鏈機制,建議將圖片保存下來直接上傳(img-cA0h3Qnw-1590984326742)(../AppData/Roaming/Typora/typora-user-images/image-20200529152434229.png)]

以下是本人項目中使用總結:

實體類

/**
 * 訂單表
 */
@Entity
@Table(name = "signedorder")
@Getter
@Setter
@NoArgsConstructor
@EntityListeners(AuditingEntityListener.class)
public class SignedOrder {
    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    @Column
    private Integer id;//id
    @CreatedDate
    @Column(updatable = false)
    private Date createTime;//創(chuàng)建時間
    @LastModifiedDate
    @Column
    private Date lastModifiedTime;//修改時間
    @ManyToOne(fetch = FetchType.EAGER, cascade = {
            CascadeType.MERGE, CascadeType.PERSIST, CascadeType.REFRESH
    })
    @JoinTable(name = "staff_signedorder", joinColumns = @JoinColumn(name =
            "signedorder_id"), inverseJoinColumns = @JoinColumn(name = "staff_id"))
    private Staff staff;//所屬用戶
    @JoinColumn(name = "industry_id")
    private Integer industryId;//行業(yè)Id
}
/**
 * 用戶表
 */
@Entity
@Table(name = "staff")
@Getter
@Setter
@NoArgsConstructor
public class Staff {
 
    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    @Column(name = "id")
    private Integer id;//id
    @Column(name = "name", length = 25)
    private String name;//姓名
    @JoinColumn(name = "city_id")
    private Integer cityId;//城市id
/**
 * 城市表
 */
@Entity
@Table(name = "city")
@Getter
@Setter
@NoArgsConstructor
@Accessors(chain = true)
public class City {
    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    @Column(name = "id")
    private Integer id;//id
    @Column(name = "name", length = 50)
    private String name;//名稱
}
//行業(yè)表和城市表一致,就不展示了

注解解釋

實體類中相關注解解釋:

  • @Entity: 對實體注釋。任何Hibernate映射對象都要有這個注釋
  • @Table: 聲明此對象映射到數(shù)據(jù)庫的數(shù)據(jù)表,該注釋不是必須的,如果沒有則系統(tǒng)使用默認值(實體的短類名)
  • @Getter、 @Setter、 @NoArgsConstructor:lombok提供注解,get、set方法及無參構造
  • @EntityListeners(AuditingEntityListener.class):加上此注解,時間注解@LastModifiedDate 和 @CreatedDate才可以生效
  • @Id: 聲明此屬性為主鍵
  • @GeneratedValue(strategy = GenerationType.IDENTITY):指定主鍵,

TABLE:使用一個特定的數(shù)據(jù)庫表格來保存主鍵;

IDENTITY:主鍵由數(shù)據(jù)庫自動生成(主要是自動增長型);

SEQUENCR:根據(jù)底層數(shù)據(jù)庫的序列來生成主鍵,條件是數(shù)據(jù)庫支持序列;

AUTO:主鍵由程序控制

  • @CreatedDate(updatable = false):創(chuàng)建時間時間字段,在insert的時候,會設置值;update時時間不變
  • @LastModifiedDate:修改時間段,update時會修改值
  • @ManyToOne(fetch = FetchType.EAGER, cascade = {CascadeType.MERGE, CascadeType.PERSIST, CascadeType.REFRESH}): 多對一,

FetchType.EAGER:立即加載, 獲取關聯(lián)實體;

CascadeType.MERGE: 級聯(lián)更新;

CascadeType.PERSIST:級聯(lián)新建;

CascadeType.REFRESH:級聯(lián)刷新

  • @JoinTable: JoinColumn:保存關聯(lián)關系的外鍵的字段;inverseJoinColumns:保存關系的另外一個外鍵字
  • @Column:用來標識實體類中屬性與數(shù)據(jù)表中字段的對應關系

測試類

@RunWith(SpringRunner.class)
@SpringBootTest
public class SprinBootMarketingsystemApplicationTests {
    @PersistenceContext//jpa的數(shù)據(jù)庫操作類
    private EntityManager entityManger;
    @Test
    public void queryDb(){
       //給參數(shù)賦值
        Integer cityId = 1;
        Integer industryId = 2;
        Integer staffId = 16;
        Date startTime = DateUtil.parse("1970-01-01");//字符串時間轉換為date類型
        Date endTime = Calendar.getInstance().getTime();//獲取系統(tǒng)當前時間裝換為date類型
        
        //創(chuàng)建SQL語句主體
        StringBuffer stringBuffer = new StringBuffer("\tSELECT\n" +
                "\tcount( * ) count,\n" +
                "\tci.NAME cityName\n" +
                "\tFROM\n" +
                "\tsignedorder s\n" +
                "\tLEFT JOIN staff_signedorder t ON s.id = t.signedorder_id\n" +
                "\tLEFT JOIN staff sta ON t.staff_id = sta.id\n" +
                "\tLEFT JOIN city ci ON sta.city_id = ci.id\n" +
                "\tWHERE\n" +
                "\t1 = 1");
         Map<String,Object> map =  new HashMap<>();
         //拼接動態(tài)參數(shù)
        if(industryId != null){
            
             /*第一種給參數(shù)賦值方式
             1代表傳進來的參數(shù)順序,給參數(shù)賦值nativeQuery.setParameter(1, industryId);
             stringBuffer.append(" and s.industryId = ?1");*/
             //industryId代表傳進來的參數(shù)名稱,給參數(shù)賦值nativeQuery.setParameter("industryId", industryId);
             stringBuffer.append(" and s.industry_id = :industryId");
             map.put("industryId",industryId);
        }
        if(cityId != null){
            stringBuffer.append(" and ci.id = :cityId");
            map.put("cityId",cityId);
        }
        if(staffId != null){
            stringBuffer.append(" and sta.id = :staffId");
            map.put("staffId",staffId);
        }
        if(startTime!=null && endTime!=null){
            //使用這種賦值方式,時間類型需要給三個參數(shù),參數(shù)名稱,參數(shù)值,特定映射的類型TemporalType.DATE
            //nativeQuery.setParameter("create_time", startTime,TemporalType.DATE);
            stringBuffer.append( " and s.create_time BETWEEN :startTime and :endTime ");
            map.put("startTime",startTime);
            map.put("endTime",endTime);
        }
        Query nativeQuery = entityManger.createNativeQuery(stringBuffer.toString());
        for (String key : map.keySet()) {
            nativeQuery.setParameter(key, map.get(key));
        }
        //三種接受返回結果方式(第一種方式)
        /*nativeQuery.unwrap(SQLQuery.class).setResultTransformer(Transformers.TO_LIST);
        List resultList1 = nativeQuery.getResultList();
        for (Object o : resultList1) {
            System.out.println(o.toString());
        }*/
       //第二種方式和第一種方式相似
        /*nativeQuery.unwrap(SQLQuery.class).setResultTransformer(Transformers.ALIAS_TO_ENTITY_MAP);
        List<Map<String, Object>> resultList = nativeQuery.getResultList();
        for (Map<String, Object> map1 :resultList
        ) {
            System.out.println(map1);
        }*/
        //第三種方式:實體類接受
        nativeQuery.unwrap(SQLQuery.class).setResultTransformer(Transformers.aliasToBean(TestVo.class));
        List<TestVo> resultList = nativeQuery.getResultList();
        for (TestVo svo:resultList
        ) {
            System.out.println(svo.toString());
        }
    }

打印結果

第一種方式打印結果

[外鏈圖片轉存失敗,源站可能有防盜鏈機制,建議將圖片保存下來直接上傳(img-iB3CVBpD-1590984326744)(ceshi1.png)]

第二種方式打印結果

[外鏈圖片轉存失敗,源站可能有防盜鏈機制,建議將圖片保存下來直接上傳(img-lHOgQW55-1590984326745)(ceshi2.png)]

第三種方式打印結果

[外鏈圖片轉存失敗,源站可能有防盜鏈機制,建議將圖片保存下來直接上傳(img-JTMsc780-1590984326747)(ceshi3.png)]

TestVo實體接收類

@Data
public class TestVo {
    private String cityName;//城市名字
    private BigInteger count;//簽單數(shù)量(必須使用BigInteger類型接受)
}

以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。

相關文章

  • 當Mybatis遇上目錄樹超全完美解決方案

    當Mybatis遇上目錄樹超全完美解決方案

    這篇文章主要介紹了當Mybatis遇上目錄樹有哪些解決方法,本文給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2021-04-04
  • 詳解poi+springmvc+springjdbc導入導出excel實例

    詳解poi+springmvc+springjdbc導入導出excel實例

    本篇文章主要介紹了poi+springmvc+springjdbc導入導出excel實例,非常具有實用價值,需要的朋友可以參考下。
    2017-01-01
  • SpringMVC下實現(xiàn)Excel文件上傳下載

    SpringMVC下實現(xiàn)Excel文件上傳下載

    這篇文章主要為大家詳細介紹了SpringMVC下實現(xiàn)Excel文件上傳下載,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2018-03-03
  • java連接數(shù)據(jù)庫知識點總結以及操作應用

    java連接數(shù)據(jù)庫知識點總結以及操作應用

    這篇文章主要給大家介紹了關于java連接數(shù)據(jù)庫知識點總結以及操作應用的相關資料, 當涉及到Java中數(shù)據(jù)庫數(shù)據(jù)處理時,我們可以利用強大的Java數(shù)據(jù)庫連接技術與各種數(shù)據(jù)庫進行交互,需要的朋友可以參考下
    2023-12-12
  • 詳解java線程的開始、暫停、繼續(xù)

    詳解java線程的開始、暫停、繼續(xù)

    本文將介紹通過線程讀取文件內(nèi)容,并且可以控制線程的開始、暫停、繼續(xù),來控制讀文件。具有一定的參考作用,下面跟著小編一起來看下吧
    2017-01-01
  • springboot后端存儲富文本內(nèi)容的思路與步驟(含圖片內(nèi)容)

    springboot后端存儲富文本內(nèi)容的思路與步驟(含圖片內(nèi)容)

    在所有的編輯器中,大概最受歡迎的就是富文本編輯器和MarkDown編輯器了,下面這篇文章主要給大家介紹了關于springboot后端存儲富文本內(nèi)容的思路與步驟的相關資料,需要的朋友可以參考下
    2023-04-04
  • Java消息隊列RabbitMQ之消息回調(diào)詳解

    Java消息隊列RabbitMQ之消息回調(diào)詳解

    這篇文章主要介紹了Java消息隊列RabbitMQ之消息回調(diào)詳解,消息回調(diào),其實就是消息確認(生產(chǎn)者推送消息成功,消費者接收消息成功)  , 對于程序來說,發(fā)送者沒法確認是否發(fā)送成功,需要的朋友可以參考下
    2023-07-07
  • java模擬實現(xiàn)斗地主發(fā)牌小程序

    java模擬實現(xiàn)斗地主發(fā)牌小程序

    這篇文章主要為大家詳細介紹了java模擬實現(xiàn)斗地主發(fā)牌小程序,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2020-04-04
  • kettle中使用js調(diào)用java類的方法

    kettle中使用js調(diào)用java類的方法

    這篇文章主要介紹了kettle中使用js調(diào)用java類的方法,本文講解了注意事項和調(diào)用語法,需要的朋友可以參考下
    2015-05-05
  • 深入理解Spring Cloud Zuul過濾器

    深入理解Spring Cloud Zuul過濾器

    這篇文章主要給大家介紹了關于Spring Cloud Zuul過濾器的相關資料,通過閱讀本文您將了解:Zuul過濾器類型與請求生命周期、如何編寫Zuul過濾器、如何禁用Zuul過濾器和Spring Cloud為Zuul編寫的過濾器及其功能,需要的朋友可以參考下。
    2017-02-02

最新評論