国产xxxx99真实实拍_久久不雅视频_高清韩国a级特黄毛片_嗯老师别我我受不了了小说

資訊專欄INFORMATION COLUMN

Spring Data JPA ——默認方法使用

melody_lql / 2900人閱讀

摘要:暫未驗證聲明主鍵。為提供一個默認的構造方法。提供一個不可為的屬性的構造方法以防止出錯。根據字段查詢并排序根據字段查詢并排序,默認是順序。統計對象數量統計對象數量這是調用接口的方法來統計對象數量。

初探 SDJ

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

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

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

JpaRepository —— JPA 協議的具體實現的接口。

    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 {}

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

為使用默認的接口方法做準備

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

// SDJ 使用 @Entity,NoSQL 使用 @Document。(暫未驗證)
@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;
    // 設定默認值。
    private Double balance = 0.0;
    // 不允許更新。
    @Column(updatable = false)
    private Date createTime = new Date();

    // 為 SDJ 提供一個默認的構造方法。
    public User() {
    }

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

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

然后創建一個簡單到可能懷疑自己人生接口,繼承自 JpaRepository。

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

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

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

試試接口默認的方法 保存 保存單個對象并返回
        // 保存單個對象并返回。
        User savedUser = userRepository.save(new User("Cciradih", "hidarichaochen@gmail.com"));

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

保存多個對象并返回
        // 保存多個對象并返回。
        List userList = new ArrayList<>();
        User newUser;
        for (int i = 0; i < 10; i++) {
            newUser = new User("Cciradih" + i, "hidarichaochen@gmail.com" + i);
            userList.add(newUser);
        }
        List savedUserList = userRepository.save(userList);

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

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

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

查詢全部對象
        // 查詢全部對象。
        List foundUserList = userRepository.findAll();

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

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

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

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

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

        User user = new User("Cciradih", "hidarichaochen@gmail.com");
        // 1.使用 Example。
        // 創建 Example。
        Example userExample = Example.of(user);
        User foundExampleUser = userRepository.findOne(userExample);
        // 2.使用 ExampleMatcher。
        // 創建 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 通過其靜態方法 Example of(T probe) 來引入一個 Example 對象。然后就通過 QueryByExampleExecutor 接口的 S findOne(Example example) 方法來查詢符合范例的對象。當然這里在通常情況下是查詢不到任何對象的,因為 User 有不同的 Id 和某種意義上講不會相同的 createTime。

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

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

關于 ExampleMatcher 更多的過濾方法可以自行參照源碼。

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

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

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

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

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

根據對象刪除單個對象
        // 根據對象刪除單個對象
        User user = new User("Cciradih", "hidarichaochen@gmail.com");
        userRepository.delete(user);

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

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

這是調用 CrudRepository 接口的 void deleteAll() 方法來根據對象刪除單個對象。

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

刪除多個對象
        // 刪除多個對象
        List userList = new ArrayList<>();
        User user;
        for (int i = 0; i < 10; i++) {
            user = new User("Cciradih" + i, "hidarichaochen@gmail.com" + i);
            userList.add(user);
        }
        userRepository.delete(userList);

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

統計對象數量
        // 統計對象數量
        long count = userRepository.count();

這是調用 CrudRepository 接口的 long count() 方法來統計對象數量。

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

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

結語

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

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

源碼倉庫

Learn/Spring Data JPA at master · cciradih/Learn

系列目錄

Spring Data JPA 詳盡指南

參考

Spring Data JPA - Reference Documentation

我的咖啡館

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

文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。

轉載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/70375.html

相關文章

  • Springboot項目搭建(四)整合MySQL數據庫(JPA

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

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

    摘要:關聯關系的關聯關系定義上,感覺并不是很靈活,姿勢也比較難找。如,定義在關聯關系上的參數可以設置級聯的相關東西。因為序列化會涉及到實體類關聯對象的獲取,會觸發所有的關聯關系。 接(4) - Database 系列. Java Persistence API,可以理解就是 Java 一個持久化標準或規范,Spring Data JPA 是對它的實現。并且提供多個 JPA 廠商適配,如 Hi...

    sutaking 評論0 收藏0
  • Spring Boot [組件學習-Spring Data JPA]

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

    andong777 評論0 收藏0
  • 【從零入門系列-2】Spring Boot 之 數據庫實體定義實現

    摘要:文章系列從零入門系列之從零入門系列之程序結構設計說明前言本篇文章開始代碼實踐,系統設計從底向上展開,因此本篇先介紹如何實現數據庫表實體類的設計實現。主鍵由數據庫自動生成主要是自動增長型主鍵由程序控制。 文章系列 【從零入門系列-0】Sprint Boot 之 Hello World 【從零入門系列-1】Sprint Boot 之 程序結構設計說明 前言 本篇文章開始代碼實踐,系統...

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

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

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

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

    Dionysus_go 評論0 收藏0

發表評論

0條評論

最新活動
閱讀需要支付1元查看
<