成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

Spring Data JPA ——默認(rèn)方法使用

melody_lql / 2902人閱讀

摘要:暫未驗(yàn)證聲明主鍵。為提供一個(gè)默認(rèn)的構(gòu)造方法。提供一個(gè)不可為的屬性的構(gòu)造方法以防止出錯(cuò)。根據(jù)字段查詢并排序根據(jù)字段查詢并排序,默認(rèn)是順序。統(tǒng)計(jì)對象數(shù)量統(tǒng)計(jì)對象數(shù)量這是調(diào)用接口的方法來統(tǒng)計(jì)對象數(shù)量。

初探 SDJ

先讓我們來鞏固一下 Java 的基礎(chǔ)知識。Java 是面向?qū)ο蟮恼Z言,目的是解決現(xiàn)實(shí)生活中的問題,可以通過使用類來包裝現(xiàn)實(shí)生活中的事物成對象、使用屬性來描述對象的特點(diǎn)并使用方法來控制對象的行為。我們在 Java 里一切的操作都是針對對象本身,這也是為什么我們需要 ORM 來操作存儲在數(shù)據(jù)庫里面的“對象”。題外話,對于類似于 Redis 這種基于 Key-Value 存儲的 NoSQL(非關(guān)系型數(shù)據(jù)庫)來說,我們并不能很顯然地觀察到存儲在里面的“對象”,因?yàn)檫@種“對象”不同于關(guān)系型數(shù)據(jù)庫里的對象,比如 MySQL。在 MySQL 中,其 Table 就對應(yīng) Java 中的類,每一條 Column 對應(yīng)的就是對象,每一個(gè)字段對應(yīng)的就是屬性,當(dāng)然 MySQL 也有自己的方法,不過不在本文討論范圍內(nèi)。我們可以很直觀地通過 SQL 的查詢語句來觀察“對象”,但在 NoSQL 中,我們只能通過序列化和反序列化來寫和讀對象。相信我,您寧愿看匯編也不想看序列化之后的對象……

為了節(jié)省時(shí)間,關(guān)于用什么和怎么來構(gòu)建基本環(huán)境不在此贅述,這里提供 SDJ 的官方指南之一供您參考,只要您能引用 SDJ 的包(org.springframework.data:spring-data-jpa:1.11.6.RELEAS)就行。

SDJ 對于應(yīng)用來說最應(yīng)該關(guān)注的就是以下幾個(gè)核心接口(按照從子類到父類的繼承順序):

JpaRepository —— JPA 協(xié)議的具體實(shí)現(xiàn)的接口。

    package org.springframework.data.jpa.repository;

    public interface JpaRepository extends PagingAndSortingRepository, QueryByExampleExecutor {}

PagingAndSortingRepository ——分頁和排序的接口。QueryByExampleExecutor ——范例查詢的接口。

    package org.springframework.data.repository;

    public interface PagingAndSortingRepository extends CrudRepository {}
    
    package org.springframework.data.repository.query;

    public interface QueryByExampleExecutor {}

CrudRepository ——通用CRUD操作的接口。

    package org.springframework.data.repository;
    
    public interface CrudRepository extends Repository {}

就如同名稱所示,我們可以根據(jù)情況繼承特定的接口,不同的接口提供不同的功能,如果我們需要分頁和排序,就繼承 PagingAndSortingRepository 接口。但為了全方位地了解 SDJ,本文使用 JpaRepository 接口,由于其位于繼承樹的最底端,可以理解成二叉樹里面的樹葉,所以可以使用包括其父類的所有未被重寫方法。當(dāng)然這樣也有一些不影響功能實(shí)現(xiàn)的矛盾點(diǎn),我們一會會見到。

為使用默認(rèn)的接口方法做準(zhǔn)備

如同之前所說,我們需要針對性地通過類來構(gòu)建一個(gè)對象。這里創(chuàng)建一個(gè)用戶類(推薦使用 Lombok 來簡化代碼)。

// SDJ 使用 @Entity,NoSQL 使用 @Document。(暫未驗(yàn)證)
@Entity
public class User {
    // 聲明主鍵。
    @Id
    // 聲明由程序控制主鍵生成策略。
    @GeneratedValue(strategy = GenerationType.AUTO)
    private Long id;
    // 約束、不可為空、長度16。
    @Column(unique = true, nullable = false, length = 16)
    private String username;
    @Column(unique = true, nullable = false)
    private String email;
    // 設(shè)定默認(rèn)值。
    private Double balance = 0.0;
    // 不允許更新。
    @Column(updatable = false)
    private Date createTime = new Date();

    // 為 SDJ 提供一個(gè)默認(rèn)的構(gòu)造方法。
    public User() {
    }

    // 提供一個(gè)不可為 null 的屬性的構(gòu)造方法以防止出錯(cuò)。
    public User(String username, String email) {
        this.username = username;
        this.email = email;
    }

    // 省略一大堆 Getter、Setter 和 ToString 方法。也可以通過 Lombok 插件以注解的方式大幅度簡化,各大 IDE 均提供!
}

然后創(chuàng)建一個(gè)簡單到可能懷疑自己人生接口,繼承自 JpaRepository。

import org.springframework.data.jpa.repository.JpaRepository;
import top.cciradih.spring.data.jpa.entity.User;

//  泛型指定了能接受的對象類型和其主鍵類型,主鍵類型在一些方法里很有用。
public interface UserRepository extends JpaRepository {
}

驚不驚喜?意不意外?我們已經(jīng)可以使用它了!什么都不用寫!

試試接口默認(rèn)的方法 保存 保存單個(gè)對象并返回
        // 保存單個(gè)對象并返回。
        User savedUser = userRepository.save(new User("Cciradih", "[email protected]"));

這是調(diào)用 CrudRepository 接口的 S save(S entity) 方法來保存并返回存儲的對象。

保存多個(gè)對象并返回
        // 保存多個(gè)對象并返回。
        List userList = new ArrayList<>();
        User newUser;
        for (int i = 0; i < 10; i++) {
            newUser = new User("Cciradih" + i, "[email protected]" + i);
            userList.add(newUser);
        }
        List savedUserList = userRepository.save(userList);

這是調(diào)用 JpaRepository 接口的 List save(Iterable entities) 方法來保存多個(gè)對象。

查詢 根據(jù)主鍵查詢單個(gè)對象
        // 根據(jù)主鍵查詢單個(gè)對象。
        User foundUser = userRepository.findOne(1L);

這是調(diào)用 CrudRepository 接口的 T findOne(ID id) 方法來查詢單個(gè)對象。

查詢?nèi)繉ο?/b>
        // 查詢?nèi)繉ο蟆?        List foundUserList = userRepository.findAll();

這是調(diào)用 JpaRepository 接口的 List findAll() 方法來查全部對象。

根據(jù) ID 字段查詢并排序
import org.springframework.data.domain.Sort;

        // 根據(jù) id 字段查詢并排序,默認(rèn)是順序(ASC)。
        List foundASCSortedUserList = userRepository.findAll(new Sort("id"));
        // 根據(jù) id 字段倒序查詢(DESC)。
        List foundDESCSortedUserList = userRepository.findAll(new Sort(Sort.Direction.DESC, "id"));

這里涉及到新的對象—— new Sort("id")。從名字可以看得出來是為了排序的,其內(nèi)部的具體邏輯在本文就不贅述。通過 JpaRepository 接口的 List findAll(Sort sort) 方法來查詢并排序。

范例查詢
import org.springframework.data.domain.Example;
import org.springframework.data.domain.ExampleMatcher;

        User user = new User("Cciradih", "[email protected]");
        // 1.使用 Example。
        // 創(chuàng)建 Example。
        Example userExample = Example.of(user);
        User foundExampleUser = userRepository.findOne(userExample);
        // 2.使用 ExampleMatcher。
        // 創(chuàng)建 ExampleMatcher。
        ExampleMatcher exampleMatcher = ExampleMatcher.matching()
                // 忽略 id 和 createTime 字段。
                .withIgnorePaths("id", "createTime")
                // 忽略大小寫。
                .withIgnoreCase()
                // 忽略為空字段。
                .withIgnoreNullValues();
        // 攜帶 ExampleMatcher。
        userExample = Example.of(user, exampleMatcher);
        User foundExampleWithExampleMatcherUser = userRepository.findOne(userExample);

這里涉及到 Example 和 ExampleMatcher 的使用。

Example 通過其靜態(tài)方法 Example of(T probe) 來引入一個(gè) Example 對象。然后就通過 QueryByExampleExecutor 接口的 S findOne(Example example) 方法來查詢符合范例的對象。當(dāng)然這里在通常情況下是查詢不到任何對象的,因?yàn)?User 有不同的 Id 和某種意義上講不會相同的 createTime。

ExampleMatcher 通過靜態(tài)方法 Example of(T probe, ExampleMatcher matcher) 攜帶一個(gè)符合的泛型對象和 ExampleMatcher 對象,這里忽略了 id 和 createTime 字段,所以能夠查找到符合條件的對象。

ExampleMatcher 是對 Example 的拓展,這也是為什么我說 SDJ 的控制細(xì)膩度依然很高的原因。

關(guān)于 ExampleMatcher 更多的過濾方法可以自行參照源碼。

分頁查詢
import org.springframework.data.domain.Page;
import org.springframework.data.domain.PageRequest;

        // 分頁查詢,從 0 頁開始查詢 5 個(gè)。
        Page foundUserPage = userRepository.findAll(new PageRequest(0, 5));
        // 分頁表。
        List content = foundUserPage.getContent();
        // 總數(shù)量。
        long totalElements = foundUserPage.getTotalElements();
        // 總頁數(shù)。
        long totalPages = foundUserPage.getTotalPages();
        // 分頁表大小。
        int size = foundUserPage.getSize();

這里涉及到新的對象—— new PageRequest(0, 5)。從名字可以看得出來是為了分頁的,其內(nèi)部的具體邏輯在本文就不贅述。通過 PagingAndSortingRepository 接口的 Page findAll(Pageable pageable) 方法來查詢 Page 對象。Page 包含了所有分頁需要的屬性,可以通過其方法來獲取。

刪除 根據(jù)主鍵刪除單個(gè)對象
        // 根據(jù)主鍵刪除單個(gè)對象
        userRepository.delete(1L);

這是調(diào)用 CrudRepository 接口的 void delete(ID id) 方法來根據(jù)主鍵刪除單個(gè)對象。

根據(jù)對象刪除單個(gè)對象
        // 根據(jù)對象刪除單個(gè)對象
        User user = new User("Cciradih", "[email protected]");
        userRepository.delete(user);

這是調(diào)用 CrudRepository 接口的 void delete(T entity) 方法來根據(jù)對象刪除單個(gè)對象。和 Example 的例子一樣,也不會刪除任何對象,請自行思考。

刪除全部
        // 刪除全部
        userRepository.deleteAll();

這是調(diào)用 CrudRepository 接口的 void deleteAll() 方法來根據(jù)對象刪除單個(gè)對象。

之前提到過的一些不影響功能實(shí)現(xiàn)的矛盾點(diǎn),在這里體現(xiàn)。由于 JpaRepository 間接繼承自 CrudRepository 所以我們可以同時(shí)使用 void deleteAllInBatch()void deleteAll() 方法,兩者的功能是一樣的,可以二者取其一。想要避免這樣的事,只能在不影響實(shí)際業(yè)務(wù)邏輯的情況下往上繼承接口。

刪除多個(gè)對象
        // 刪除多個(gè)對象
        List userList = new ArrayList<>();
        User user;
        for (int i = 0; i < 10; i++) {
            user = new User("Cciradih" + i, "[email protected]" + i);
            userList.add(user);
        }
        userRepository.delete(userList);

這是調(diào)用 CrudRepository 接口的 void delete(Iterable entities) 方法來刪除多個(gè)對象。和 Example 的例子一樣,也不會刪除任何對象,請自行思考。

統(tǒng)計(jì)對象數(shù)量
        // 統(tǒng)計(jì)對象數(shù)量
        long count = userRepository.count();

這是調(diào)用 CrudRepository 接口的 long count() 方法來統(tǒng)計(jì)對象數(shù)量。

根據(jù)主鍵判斷對象是否存在
        // 判斷對象是否存在
        boolean exists = userRepository.exists(1L);

這是調(diào)用 CrudRepository 接口的 boolean exists(ID id) 方法來根據(jù)主鍵判斷對象是否存在。

結(jié)語

我盡可能地列舉了默認(rèn)方法,但為了行文方便,其中有些方法是可以配合使用的,比如 JpaRepository 接口的 List findAll(Example example, Sort sort),可以同時(shí)傳入 Example 和 Sort 對象進(jìn)行查詢,請自行參照源碼使用。

我希望您能夠通過此文對 SDJ 的默認(rèn)方法達(dá)到“知其然”的程度,在軟件開發(fā)中,我認(rèn)為需要從“知其然”到”知其所以然“再到”造其所以然“的地步,但路需要一步一步走。

源碼倉庫

Learn/Spring Data JPA at master · cciradih/Learn

系列目錄

Spring Data JPA 詳盡指南

參考

Spring Data JPA - Reference Documentation

我的咖啡館

如果您對本文有什么建議或者問題,歡迎您來我的咖啡館坐坐338147322。

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/70375.html

相關(guān)文章

  • Springboot項(xiàng)目搭建(四)整合MySQL數(shù)據(jù)庫(JPA

    摘要:整合數(shù)據(jù)庫一文件添加依賴二配置文件主參數(shù)指定指定數(shù)據(jù)源用戶名指定數(shù)據(jù)源密碼指定當(dāng)使用內(nèi)嵌數(shù)據(jù)庫時(shí),默認(rèn)是,否則為是否開啟的,默認(rèn)為參考建議配置屬性之三配置實(shí)體類主鍵主鍵自增四實(shí)現(xiàn)單表操作此處泛型為映射類型 springboot整合MySQL數(shù)據(jù)庫(JPA) 一、POM文件添加依賴 org.springframework.boot spring-boot-starte...

    hikui 評論0 收藏0
  • Spring Boot QuickStart (5) - Spring Data JPA

    摘要:關(guān)聯(lián)關(guān)系的關(guān)聯(lián)關(guān)系定義上,感覺并不是很靈活,姿勢也比較難找。如,定義在關(guān)聯(lián)關(guān)系上的參數(shù)可以設(shè)置級聯(lián)的相關(guān)東西。因?yàn)樾蛄谢瘯婕暗綄?shí)體類關(guān)聯(lián)對象的獲取,會觸發(fā)所有的關(guān)聯(lián)關(guān)系。 接(4) - Database 系列. Java Persistence API,可以理解就是 Java 一個(gè)持久化標(biāo)準(zhǔn)或規(guī)范,Spring Data JPA 是對它的實(shí)現(xiàn)。并且提供多個(gè) JPA 廠商適配,如 Hi...

    sutaking 評論0 收藏0
  • Spring Boot [組件學(xué)習(xí)-Spring Data JPA]

    摘要:與的關(guān)系是什么是官方提出的持久化規(guī)范。它為開發(fā)人員提供了一種對象關(guān)聯(lián)映射工具來管理應(yīng)用中的關(guān)系數(shù)據(jù)。他的出現(xiàn)主要是為了簡化現(xiàn)有的持久化開發(fā)工作和整合技術(shù),結(jié)束現(xiàn)在,,等框架各自為營的局面。定義了在對數(shù)據(jù)庫中的對象處理查詢和事務(wù)運(yùn)行時(shí)的的。 導(dǎo)讀: 在上篇文章中對Spring MVC常用的一些注解做了簡要的說明,在這篇文章中主要對Spring Data JPA 做一個(gè)簡要的說明,并附有一...

    andong777 評論0 收藏0
  • 【從零入門系列-2】Spring Boot 之 數(shù)據(jù)庫實(shí)體定義實(shí)現(xiàn)

    摘要:文章系列從零入門系列之從零入門系列之程序結(jié)構(gòu)設(shè)計(jì)說明前言本篇文章開始代碼實(shí)踐,系統(tǒng)設(shè)計(jì)從底向上展開,因此本篇先介紹如何實(shí)現(xiàn)數(shù)據(jù)庫表實(shí)體類的設(shè)計(jì)實(shí)現(xiàn)。主鍵由數(shù)據(jù)庫自動(dòng)生成主要是自動(dòng)增長型主鍵由程序控制。 文章系列 【從零入門系列-0】Sprint Boot 之 Hello World 【從零入門系列-1】Sprint Boot 之 程序結(jié)構(gòu)設(shè)計(jì)說明 前言 本篇文章開始代碼實(shí)踐,系統(tǒng)...

    nemo 評論0 收藏0
  • 貓頭鷹的深夜翻譯:為什么要使用Spring Boot?

    摘要:初次使用的人往往會困惑,不知道該使用哪種方法。目前來說,團(tuán)隊(duì)推薦使用基于的方法來提供更高的靈活性。配置,從而在應(yīng)用啟動(dòng)時(shí)執(zhí)行腳本來初始化數(shù)據(jù)庫。目前為止我們沒有任何消息需要配置,所以只在文件夾中創(chuàng)建一個(gè)空的文件。將配置為,它包含的上下文。 前言 spring是一個(gè)用于創(chuàng)建web和企業(yè)應(yīng)用的一個(gè)很流行的框架。和別的只關(guān)注于一點(diǎn)的框架不同,Spring框架通過投資并組合項(xiàng)目提供了大量的功能...

    Jaden 評論0 收藏0
  • 一起來學(xué)SpringBoot | 第六篇:整合SpringDataJpa

    摘要:忽略該字段的映射省略創(chuàng)建數(shù)據(jù)訪問層接口,需要繼承,第一個(gè)泛型參數(shù)是實(shí)體對象的名稱,第二個(gè)是主鍵類型。 SpringBoot 是為了簡化 Spring 應(yīng)用的創(chuàng)建、運(yùn)行、調(diào)試、部署等一系列問題而誕生的產(chǎn)物,自動(dòng)裝配的特性讓我們可以更好的關(guān)注業(yè)務(wù)本身而不是外部的XML配置,我們只需遵循規(guī)范,引入相關(guān)的依賴就可以輕易的搭建出一個(gè) WEB 工程 上一篇介紹了Spring JdbcTempl...

    Dionysus_go 評論0 收藏0

發(fā)表評論

0條評論

最新活動(dòng)
閱讀需要支付1元查看
<