Spring Boot 整合 JPA 实现数据持久化

news2025/2/12 22:16:26

目录

前言

一、JPA 核心概念与实体映射

1. 什么是 JPA?

2. JPA 的主要组件

3. 实体映射

4. 常见的字段映射策略

二、Repository 接口与自定义查询

1. 什么是 Repository 接口?

2. 动态查询方法

3. 自定义查询

4. 分页与排序

三、实战案例:完整数据持久化示例

1. 创建一个简单的用户管理系统

2. 测试接口


前言

在现代企业级应用开发中,数据持久化是不可或缺的一部分。Spring Boot 提供了对 JPA(Java Persistence API)的强大支持,使得数据库操作变得更加简单和高效。本文将从以下几个方面详细讲解如何在 Spring Boot 中整合 JPA 实现数据持久化:

  1. JPA 核心概念与实体映射
  2. Repository 接口与自定义查询

通过本文的学习,你将能够掌握 Spring Boot 中 JPA 的基本使用方法,并能够根据实际需求进行灵活的扩展。


一、JPA 核心概念与实体映射

1. 什么是 JPA?

JPA(Java Persistence API)是 Java EE 平台中用于对象关系映射(ORM)的标准 API。它允许开发者通过 Java 类来表示数据库中的表,并通过 JPA 提供的接口和注解来执行 CRUD(增删改查)操作。

2. JPA 的主要组件
  • EntityManager:负责管理实体对象的生命周期,并提供 CRUD 操作的方法。
  • Persistence Unit:一组相关的实体类和配置信息的集合。
  • Entity:表示数据库表的 Java 类。
3. 实体映射

在 JPA 中,实体类通过注解来映射到数据库表。以下是常用的注解:

  • @Entity:标识这是一个实体类。
  • @Table:指定实体类对应的数据库表名。
  • @Id:标识实体类的主键字段。
  • @GeneratedValue:指定主键的生成策略。
  • @Column:指定字段对应的数据库列名、长度等属性。

示例代码:

import jakarta.persistence.*; 
 
@Entity 
@Table(name = "users")
public class User {
 
    @Id 
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    private Long id;
 
    @Column(name = "username", length = 50, nullable = false)
    private String username;
 
    @Column(name = "email", unique = true)
    private String email;
 
    // Getter and Setter methods 
}
4. 常见的字段映射策略
  • 基本类型映射:如 StringIntegerLong 等。
  • 关联关系映射
    • 一对一(OneToOne):通过 @OneToOne 注解实现。
    • 一对多(OneToMany):通过 @OneToMany 注解实现。
    • 多对一(ManyToOne):通过 @ManyToOne 注解实现。
    • 多对多(ManyToMany):通过 @ManyToMany 注解实现。

示例代码(一对多关系):

import jakarta.persistence.*; 
 
@Entity 
@Table(name = "posts")
public class Post {
 
    @Id 
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    private Long id;
 
    @Column(name = "title", nullable = false)
    private String title;
 
    @Column(name = "content")
    private String content;
 
    @ManyToOne 
    @JoinColumn(name = "user_id")
    private User user;
 
    // Getter and Setter methods 
}

二、Repository 接口与自定义查询

1. 什么是 Repository 接口?

Spring Data JPA 提供了一个 Repository 接口,用于简化数据访问层的开发。通过继承 JpaRepositoryCrudRepository,我们可以快速获得 CRUD 操作的能力。

示例代码:

import org.springframework.data.jpa.repository.JpaRepository; 
import org.springframework.stereotype.Repository; 
 
@Repository 
public interface UserRepository extends JpaRepository<User, Long> {
}

  • JpaRepository:继承了 CrudRepository PagingAndSortingRepository,提供了基本的 CRUD 操作和分页支持。
  • UserRepository:定义了一个针对 User 实体的仓库接口。
2. 动态查询方法

Spring Data JPA 支持通过方法名动态生成查询语句。例如:

public interface UserRepository extends JpaRepository<User, Long> {
 
    List<User> findByUsername(String username);
 
    List<User> findByEmailContaining(String keyword);
 
    List<User> findByUsernameAndEmail(String username, String email);
}

  • findByUsername:根据用户名查询用户。
  • findByEmailContaining:根据邮箱包含关键字查询用户。
  • findByUsernameAndEmail:根据用户名和邮箱联合查询用户。
3. 自定义查询

对于复杂的查询需求,可以通过 @Query 注解来自定义 JPQL(Java Persistence Query Language)或原生 SQL 查询。

示例代码:

import org.springframework.data.jpa.repository.JpaRepository; 
import org.springframework.data.jpa.repository.Query; 
import org.springframework.data.repository.query.Param; 
import org.springframework.stereotype.Repository; 
 
import java.util.List; 
 
@Repository 
public interface UserRepository extends JpaRepository<User, Long> {
 
    @Query("SELECT u FROM User u WHERE u.username  LIKE %:keyword%")
    List<User> findByUsernameLike(@Param("keyword") String keyword);
 
    @Query(value = "SELECT * FROM users WHERE email LIKE %?1%", nativeQuery = true)
    List<User> findByEmailLike(String keyword);
}
  • findByUsernameLike:使用 JPQL 查询用户名包含关键字的用户。
  • findByEmailLike:使用原生 SQL 查询邮箱包含关键字的用户。
4. 分页与排序

Spring Data JPA 提供了 Pageable 接口来支持分页和排序。

示例代码:

import org.springframework.data.domain.Page; 
import org.springframework.data.domain.Pageable; 
 
public interface UserRepository extends JpaRepository<User, Long> {
 
    Page<User> findAll(Pageable pageable);
 
    Page<User> findByUsernameContaining(String keyword, Pageable pageable);
}

三、实战案例:完整数据持久化示例

1. 创建一个简单的用户管理系统

实体类(User.java ):

import jakarta.persistence.*; 
 
@Entity 
@Table(name = "users")
public class User {
 
    @Id 
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    private Long id;
 
    @Column(name = "username", length = 50, nullable = false)
    private String username;
 
    @Column(name = "email", unique = true)
    private String email;
 
    // Getter and Setter methods 
}

Repository 接口(UserRepository.java ):

import org.springframework.data.jpa.repository.JpaRepository; 
import org.springframework.data.jpa.repository.Query; 
import org.springframework.data.repository.query.Param; 
import org.springframework.stereotype.Repository; 
 
import java.util.List; 
 
@Repository 
public interface UserRepository extends JpaRepository<User, Long> {
 
    List<User> findByUsername(String username);
 
    @Query("SELECT u FROM User u WHERE u.email  LIKE %:keyword%")
    List<User> findByEmailLike(@Param("keyword") String keyword);
}

 Service 层(UserService.java ):

import org.springframework.beans.factory.annotation.Autowired; 
import org.springframework.stereotype.Service; 
 
import java.util.List; 
 
@Service 
public class UserService {
 
    @Autowired 
    private UserRepository userRepository;
 
    public List<User> findAllUsers() {
        return userRepository.findAll(); 
    }
 
    public User saveUser(User user) {
        return userRepository.save(user); 
    }
 
    public List<User> findByUsername(String username) {
        return userRepository.findByUsername(username); 
    }
 
    public List<User> findByEmailLike(String keyword) {
        return userRepository.findByEmailLike(keyword); 
    }
}

Controller 层(UserController.java ):

import org.springframework.beans.factory.annotation.Autowired; 
import org.springframework.web.bind.annotation.*; 
 
import java.util.List; 
 
@RestController 
@RequestMapping("/api/users")
public class UserController {
 
    @Autowired 
    private UserService userService;
 
    @GetMapping 
    public List<User> getAllUsers() {
        return userService.findAllUsers(); 
    }
 
    @PostMapping 
    public User createUser(@RequestBody User user) {
        return userService.saveUser(user); 
    }
 
    @GetMapping("/username/{username}")
    public List<User> findByUsername(@PathVariable String username) {
        return userService.findByUsername(username); 
    }
 
    @GetMapping("/email/{keyword}")
    public List<User> findByEmailLike(@PathVariable String keyword) {
        return userService.findByEmailLike(keyword); 
    }
}

2. 测试接口

启动应用后,可以通过 Postman 或 Swagger UI 测试以下接口:

  • GET http://localhost:8080/api/users:获取所有用户。
  • POST http://localhost:8080/api/users:创建新用户。
  • GET http://localhost:8080/api/users/username/{username}:根据用户名查询用户。
  • GET http://localhost:8080/api/users/email/{keyword}:根据邮箱关键字查询用户。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2296485.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【零基础学Mysql】常用函数讲解,提升数据操作效率的利器

以耳倾听世间繁华&#xff0c;以语表达心中所想 大家好,我是whisperrrr. 前言&#xff1a; 大家好&#xff0c;我是你们的朋友whisrrr。在日常工作中&#xff0c;MySQL作为一款广泛使用的开源关系型数据库&#xff0c;其强大的功能为我们提供了便捷的数据存储和管理手段。而在…

防火墙安全综合实验

防火墙安全综合实验 一、拓扑信息 二、需求及配置 实验步骤 需求一&#xff1a;根据下表&#xff0c;完成相关配置 设备接口VLAN接口类型SW2GE0/0/2VLAN 10AccessGE0/0/3VLAN 20AccessGE0/0/1VLAN List&#xff1a;10 20Trunk 1、创建vlan10和vlan20 2、将接口划分到对应…

RabbitMQ 消息顺序性保证

方式一&#xff1a;Consumer设置exclusive 注意条件 作用于basic.consume不支持quorum queue 当同时有A、B两个消费者调用basic.consume方法消费&#xff0c;并将exclusive设置为true时&#xff0c;第二个消费者会抛出异常&#xff1a; com.rabbitmq.client.AlreadyClosedEx…

DeepSeek R1 简单指南:架构、训练、本地部署和硬件要求

DeepSeek R1 简单指南&#xff1a;架构、训练、本地部署和硬件要求 DeepSeek 的 LLM 推理新方法 DeepSeek 推出了一种创新方法&#xff0c;通过强化学习 (RL) 来提高大型语言模型 (LLM) 的推理能力&#xff0c;其最新论文 DeepSeek-R1 对此进行了详细介绍。这项研究代表了我们…

1.攻防世界 unserialize3(wakeup()魔术方法、反序列化工作原理)

进入题目页面如下 直接开审 <?php // 定义一个名为 xctf 的类 class xctf {// 声明一个公共属性 $flag&#xff0c;初始值为字符串 111public $flag 111;// 定义一个魔术方法 __wakeup()// 当对象被反序列化时&#xff0c;__wakeup() 方法会自动调用public function __wa…

【R语言】卡方检验

一、定义 卡方检验是用来检验样本观测次数与理论或总体次数之间差异性的推断性统计方法&#xff0c;其原理是比较观测值与理论值之间的差异。两者之间的差异越小&#xff0c;检验的结果越不容易达到显著水平&#xff1b;反之&#xff0c;检验结果越可能达到显著水平。 二、用…

2025.2.9机器学习笔记:PINN文献阅读

2025.2.9周报 文献阅读题目信息摘要Abstract创新点网络架构实验结论缺点以及后续展望 文献阅读 题目信息 题目&#xff1a; GPT-PINN:Generative Pre-Trained Physics-Informed Neural Networks toward non-intrusive Meta-learning of parametric PDEs期刊&#xff1a; Fini…

JVM(Java 虚拟机)

Java语言的解释性和编译性&#xff08;通过JVM 的执行引擎&#xff09; Java 代码&#xff08;.java 文件&#xff09;要先使用 javac 编译器编译为 .class 文件&#xff08;字节码&#xff09;&#xff0c;紧接着再通过JVM 的执行引擎&#xff08;Execution Engine&#xff09…

利用二分法进行 SQL 盲注

什么是sql注入&#xff1f; SQL 注入&#xff08;SQL Injection&#xff09;是一种常见的 Web 安全漏洞&#xff0c;攻击者可以通过构造恶意 SQL 语句来访问数据库中的敏感信息。在某些情况下&#xff0c;服务器不会直接返回查询结果&#xff0c;而是通过布尔值&#xff08;Tr…

大模型数据集全面整理:444个数据集下载地址

本文针对Datasets for Large Language Models: A Comprehensive Survey 中的 444 个数据集&#xff08;涵盖8种语言类别和32个领域&#xff09;进行完整下载地址整理收集。 2024-02-28&#xff0c;由杨刘、曹家欢、刘崇宇、丁凯、金连文等作者编写&#xff0c;深入探讨了大型语…

Linux 创建进程 fork()、vfork() 与进程管理

Linux 创建进程 fork、vfork、进程管理 一、Linux的0号、1号、2号进程二、Linux的进程标识三、fork() 函数1、基本概念2、函数特点3、用法以及应用场景&#xff08;1&#xff09;父子进程执行不同的代码&#xff08;2&#xff09;进程执行另一个程序 4、工作原理 四、vfork() 函…

2025web寒假作业二

一、整体功能概述 该代码构建了一个简单的后台管理系统界面&#xff0c;主要包含左侧导航栏和右侧内容区域。左侧导航栏有 logo、管理员头像、导航菜单和安全退出按钮&#xff1b;右侧内容区域包括页头、用户信息管理内容&#xff08;含搜索框和用户数据表格&#xff09;以及页…

鸿蒙NEXT API使用指导之文件压缩和邮件创建

鸿蒙NEXT API 使用指导 一、前言二、邮件创建1、拉起垂类应用2、 UIAbilityContext.startAbilityByType 原型2.1、wantParam2.2、abilityStartCallback 与 callback 3、拉起邮箱类应用3.1、单纯拉起邮箱应用3.2、传入带附件的邮件 三、压缩文件1、认识 zlib2、压缩处理2.1、单文…

javaEE-10.CSS入门

目录 一.什么是CSS ​编辑二.语法规则: 三.使用方式 1.行内样式: 2.内部样式: 3.外部样式: 空格规范 : 四.CSS选择器类型 1.标签选择器 2.类选择器 3.ID选择器 4.通配符选择器 5.复合选择器 五.常用的CSS样式 1.color:设置字体颜色 2.font-size:设置字体大小 3…

Spring Boot牵手Redisson:分布式锁实战秘籍

一、引言 在当今的分布式系统架构中,随着业务规模的不断扩大和系统复杂度的日益增加,如何确保多个服务节点之间的数据一致性和操作的原子性成为了一个至关重要的问题。在单机环境下,我们可以轻松地使用线程锁或进程锁来控制对共享资源的访问,但在分布式系统中,由于各个服务…

制药行业 BI 可视化数据分析方案

一、行业背景 随着医药行业数字化转型的深入&#xff0c;企业积累了海量的数据&#xff0c;包括销售数据、生产数据、研发数据、市场数据等。如何利用这些数据&#xff0c;挖掘其价值&#xff0c;为企业决策提供支持&#xff0c;成为医药企业面临的重大挑战。在当今竞争激烈的…

[学习笔记] Kotlin Compose-Multiplatform

Compose-Multiplatform 原文&#xff1a;https://github.com/zimoyin/StudyNotes-master/blob/master/compose-multiplatform/compose.md Compose Multiplatform 是 JetBrains 为桌面平台&#xff08;macOS&#xff0c;Linux&#xff0c;Windows&#xff09;和Web编写Kotlin UI…

Golang 并发机制-7:sync.Once实战应用指南

Go的并发模型是其突出的特性之一&#xff0c;但强大的功能也带来了巨大的责任。sync.Once是由Go的sync包提供的同步原语。它的目的是确保一段代码只执行一次&#xff0c;而不管有多少协程试图执行它。这听起来可能很简单&#xff0c;但它改变了并发环境中管理一次性操作的规则。…

【AI实践】Cursor上手-跑通Hello World和时间管理功能

背景 学习目的&#xff1a;熟悉Cursor使用环境&#xff0c;跑通基本开发链路。 本人背景&#xff1a;安卓开发不熟悉&#xff0c;了解科技软硬件常识 实践 基础操作 1&#xff0c;下载安装安卓Android Studio 创建一个empty project 工程&#xff0c;名称为helloworld 2&am…

【多模态大模型】系列4:目标检测(ViLD、GLIP)

目录 1 ViLD2 GLIP 1 ViLD OPEN-VOCABULARY OBJECT DETECTION VIA VISION AND LANGUAGE KNOWLEDGE DISTILLATION 从标题就能看出来&#xff0c;作者是把CLIP模型当成一个Teacher&#xff0c;去蒸馏他自己的网络&#xff0c;从而能Zero Shot去做目标检测。 现在的目标检测数据…