摘要:暫未驗證聲明主鍵。為提供一個默認(rèn)的構(gòu)造方法。提供一個不可為的屬性的構(gòu)造方法以防止出錯。根據(jù)字段查詢并排序根據(jù)字段查詢并排序,默認(rèn)是順序。統(tǒng)計對象數(shù)量統(tǒng)計對象數(shù)量這是調(diào)用接口的方法來統(tǒng)計對象數(shù)量。
初探 SDJ
先讓我們來鞏固一下 Java 的基礎(chǔ)知識。Java 是面向?qū)ο蟮恼Z言,目的是解決現(xiàn)實生活中的問題,可以通過使用類來包裝現(xiàn)實生活中的事物成對象、使用屬性來描述對象的特點(diǎn)并使用方法來控制對象的行為。我們在 Java 里一切的操作都是針對對象本身,這也是為什么我們需要 ORM 來操作存儲在數(shù)據(jù)庫里面的“對象”。題外話,對于類似于 Redis 這種基于 Key-Value 存儲的 NoSQL(非關(guān)系型數(shù)據(jù)庫)來說,我們并不能很顯然地觀察到存儲在里面的“對象”,因為這種“對象”不同于關(guān)系型數(shù)據(jù)庫里的對象,比如 MySQL。在 MySQL 中,其 Table 就對應(yīng) Java 中的類,每一條 Column 對應(yīng)的就是對象,每一個字段對應(yīng)的就是屬性,當(dāng)然 MySQL 也有自己的方法,不過不在本文討論范圍內(nèi)。我們可以很直觀地通過 SQL 的查詢語句來觀察“對象”,但在 NoSQL 中,我們只能通過序列化和反序列化來寫和讀對象。相信我,您寧愿看匯編也不想看序列化之后的對象……
為了節(jié)省時間,關(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)注的就是以下幾個核心接口(按照從子類到父類的繼承順序):
JpaRepository —— JPA 協(xié)議的具體實現(xiàn)的接口。
package org.springframework.data.jpa.repository; public interface JpaRepositoryextends PagingAndSortingRepository , QueryByExampleExecutor {}
PagingAndSortingRepository ——分頁和排序的接口。QueryByExampleExecutor ——范例查詢的接口。
package org.springframework.data.repository; public interface PagingAndSortingRepositoryextends CrudRepository {} package org.springframework.data.repository.query; public interface QueryByExampleExecutor {}
CrudRepository ——通用CRUD操作的接口。
package org.springframework.data.repository; public interface CrudRepositoryextends Repository {}
就如同名稱所示,我們可以根據(jù)情況繼承特定的接口,不同的接口提供不同的功能,如果我們需要分頁和排序,就繼承 PagingAndSortingRepository 接口。但為了全方位地了解 SDJ,本文使用 JpaRepository 接口,由于其位于繼承樹的最底端,可以理解成二叉樹里面的樹葉,所以可以使用包括其父類的所有未被重寫方法。當(dāng)然這樣也有一些不影響功能實現(xiàn)的矛盾點(diǎn),我們一會會見到。
為使用默認(rèn)的接口方法做準(zhǔn)備如同之前所說,我們需要針對性地通過類來構(gòu)建一個對象。這里創(chuàng)建一個用戶類(推薦使用 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; // 設(shè)定默認(rèn)值。 private Double balance = 0.0; // 不允許更新。 @Column(updatable = false) private Date createTime = new Date(); // 為 SDJ 提供一個默認(rèn)的構(gòu)造方法。 public User() { } // 提供一個不可為 null 的屬性的構(gòu)造方法以防止出錯。 public User(String username, String email) { this.username = username; this.email = email; } // 省略一大堆 Getter、Setter 和 ToString 方法。也可以通過 Lombok 插件以注解的方式大幅度簡化,各大 IDE 均提供! }
然后創(chuàng)建一個簡單到可能懷疑自己人生接口,繼承自 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)的方法 保存 保存單個對象并返回// 保存單個對象并返回。 User savedUser = userRepository.save(new User("Cciradih", "hidarichaochen@gmail.com"));
這是調(diào)用 CrudRepository S save(S entity) 方法來保存并返回存儲的對象。
// 保存多個對象并返回。 ListuserList = 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);
這是調(diào)用 JpaRepository List 方法來保存多個對象。 save(Iterable entities)
// 根據(jù)主鍵查詢單個對象。 User foundUser = userRepository.findOne(1L);
這是調(diào)用 CrudRepository
// 查詢?nèi)繉ο蟆? ListfoundUserList = userRepository.findAll();
這是調(diào)用 JpaRepository
import org.springframework.data.domain.Sort; // 根據(jù) id 字段查詢并排序,默認(rèn)是順序(ASC)。 ListfoundASCSortedUserList = userRepository.findAll(new Sort("id")); // 根據(jù) id 字段倒序查詢(DESC)。 List foundDESCSortedUserList = userRepository.findAll(new Sort(Sort.Direction.DESC, "id"));
這里涉及到新的對象—— new Sort("id")。從名字可以看得出來是為了排序的,其內(nèi)部的具體邏輯在本文就不贅述。通過 JpaRepository
import org.springframework.data.domain.Example; import org.springframework.data.domain.ExampleMatcher; User user = new User("Cciradih", "hidarichaochen@gmail.com"); // 1.使用 Example。 // 創(chuàng)建 Example。 ExampleuserExample = 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)方法 S findOne(Example 方法來查詢符合范例的對象。當(dāng)然這里在通常情況下是查詢不到任何對象的,因為 User 有不同的 Id 和某種意義上講不會相同的 createTime。 example)
ExampleMatcher 通過靜態(tài)方法
ExampleMatcher 是對 Example 的拓展,這也是為什么我說 SDJ 的控制細(xì)膩度依然很高的原因。
關(guān)于 ExampleMatcher 更多的過濾方法可以自行參照源碼。
分頁查詢import org.springframework.data.domain.Page; import org.springframework.data.domain.PageRequest; // 分頁查詢,從 0 頁開始查詢 5 個。 PagefoundUserPage = 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
// 根據(jù)主鍵刪除單個對象 userRepository.delete(1L);
這是調(diào)用 CrudRepository
// 根據(jù)對象刪除單個對象 User user = new User("Cciradih", "hidarichaochen@gmail.com"); userRepository.delete(user);
這是調(diào)用 CrudRepository
// 刪除全部 userRepository.deleteAll();
這是調(diào)用 CrudRepository
之前提到過的一些不影響功能實現(xiàn)的矛盾點(diǎn),在這里體現(xiàn)。由于 JpaRepository
// 刪除多個對象 ListuserList = 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);
這是調(diào)用 CrudRepository
// 統(tǒng)計對象數(shù)量 long count = userRepository.count();
這是調(diào)用 CrudRepository
// 判斷對象是否存在 boolean exists = userRepository.exists(1L);
這是調(diào)用 CrudRepository
我盡可能地列舉了默認(rèn)方法,但為了行文方便,其中有些方法是可以配合使用的,比如 JpaRepository List,可以同時傳入 Example 和 Sort 對象進(jìn)行查詢,請自行參照源碼使用。 findAll(Example example, Sort sort)
我希望您能夠通過此文對 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://specialneedsforspecialkids.com/yun/70375.html
摘要:整合數(shù)據(jù)庫一文件添加依賴二配置文件主參數(shù)指定指定數(shù)據(jù)源用戶名指定數(shù)據(jù)源密碼指定當(dāng)使用內(nèi)嵌數(shù)據(jù)庫時,默認(rèn)是,否則為是否開啟的,默認(rèn)為參考建議配置屬性之三配置實體類主鍵主鍵自增四實現(xiàn)單表操作此處泛型為映射類型 springboot整合MySQL數(shù)據(jù)庫(JPA) 一、POM文件添加依賴 org.springframework.boot spring-boot-starte...
摘要:關(guān)聯(lián)關(guān)系的關(guān)聯(lián)關(guān)系定義上,感覺并不是很靈活,姿勢也比較難找。如,定義在關(guān)聯(lián)關(guān)系上的參數(shù)可以設(shè)置級聯(lián)的相關(guān)東西。因為序列化會涉及到實體類關(guān)聯(lián)對象的獲取,會觸發(fā)所有的關(guān)聯(lián)關(guān)系。 接(4) - Database 系列. Java Persistence API,可以理解就是 Java 一個持久化標(biāo)準(zhǔn)或規(guī)范,Spring Data JPA 是對它的實現(xiàn)。并且提供多個 JPA 廠商適配,如 Hi...
摘要:與的關(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)行時的的。 導(dǎo)讀: 在上篇文章中對Spring MVC常用的一些注解做了簡要的說明,在這篇文章中主要對Spring Data JPA 做一個簡要的說明,并附有一...
摘要:文章系列從零入門系列之從零入門系列之程序結(jié)構(gòu)設(shè)計說明前言本篇文章開始代碼實踐,系統(tǒng)設(shè)計從底向上展開,因此本篇先介紹如何實現(xiàn)數(shù)據(jù)庫表實體類的設(shè)計實現(xiàn)。主鍵由數(shù)據(jù)庫自動生成主要是自動增長型主鍵由程序控制。 文章系列 【從零入門系列-0】Sprint Boot 之 Hello World 【從零入門系列-1】Sprint Boot 之 程序結(jié)構(gòu)設(shè)計說明 前言 本篇文章開始代碼實踐,系統(tǒng)...
摘要:初次使用的人往往會困惑,不知道該使用哪種方法。目前來說,團(tuán)隊推薦使用基于的方法來提供更高的靈活性。配置,從而在應(yīng)用啟動時執(zhí)行腳本來初始化數(shù)據(jù)庫。目前為止我們沒有任何消息需要配置,所以只在文件夾中創(chuàng)建一個空的文件。將配置為,它包含的上下文。 前言 spring是一個用于創(chuàng)建web和企業(yè)應(yīng)用的一個很流行的框架。和別的只關(guān)注于一點(diǎn)的框架不同,Spring框架通過投資并組合項目提供了大量的功能...
摘要:忽略該字段的映射省略創(chuàng)建數(shù)據(jù)訪問層接口,需要繼承,第一個泛型參數(shù)是實體對象的名稱,第二個是主鍵類型。 SpringBoot 是為了簡化 Spring 應(yīng)用的創(chuàng)建、運(yùn)行、調(diào)試、部署等一系列問題而誕生的產(chǎn)物,自動裝配的特性讓我們可以更好的關(guān)注業(yè)務(wù)本身而不是外部的XML配置,我們只需遵循規(guī)范,引入相關(guān)的依賴就可以輕易的搭建出一個 WEB 工程 上一篇介紹了Spring JdbcTempl...
閱讀 3650·2021-10-12 10:11
閱讀 1013·2021-09-22 15:42
閱讀 3465·2019-08-30 13:06
閱讀 907·2019-08-29 17:05
閱讀 1651·2019-08-29 12:21
閱讀 2378·2019-08-29 11:31
閱讀 1136·2019-08-23 18:37
閱讀 1257·2019-08-23 14:58