Spring基础分析08-集成JPA/Hibernate进行ORM操作

大家好,今天和大家一起分享一下Spring集成JPAHibernate进行ORM操作的流程~

JPA(Java Persistence API)作为Java EE标准的一部分,提供了统一的API来管理实体类和持久化上下文;Hibernate则是最流行的JPA实现之一,以其灵活性和强大功能著称。Spring框架通过其数据访问抽象层,为JPA/Hibernate提供了卓越的支持,使得ORM操作变得更加简单、直观且高效。

ORM的基本概念

对象关系映射是一种编程技术,用于在面向对象的数据模型和关系型数据库之间建立映射关系。通过ORM工具,我们可以像操作普通Java对象一样对数据库中的记录进行增删改查等操作,而不需要关心底层SQL语句的具体实现。

JPA与Hibernate

JPA是Java平台上的持久性标准,定义了一组接口和注解来描述实体类及其关系。Hibernate是JPA的一个开源实现,除了遵循JPA规范外,还提供了许多额外的功能,如二级缓存、查询语言HQL等。选择JPA或Hibernate取决于项目的具体需求和技术栈偏好。

Spring集成JPA/Hibernate的基础

Spring Data JPA是Spring Data项目的一部分,旨在简化JPA的使用。它提供了一套通用的CRUD仓库接口和查询方法,让我们可以专注于业务逻辑而不是繁琐的数据访问细节。此外,Spring Data JPA还支持自定义查询方法、分页、排序等功能,极大地提高了开发效率。

配置Spring Boot项目

为了快速开始一个新的Spring Boot项目,我们可以使用Spring Initializr生成基础代码结构。在这个过程中,选择Web、JPA、Thymeleaf(或其他模板引擎)、H2(内存数据库)等依赖项。接下来,在application.properties文件中配置数据库连接信息:


spring.datasource.url=jdbc:h2:mem:testdb
spring.datasource.driverClassName=org.h2.Driver
spring.datasource.username=sa
spring.datasource.password=password
spring.jpa.database-platform=org.hibernate.dialect.H2Dialect
spring.h2.console.enabled=true

对于实际生产环境,建议使用MySQL、PostgreSQL等外部数据库,并相应调整配置参数。

实体类定义

基本注解

在定义实体类时,我们需要使用一系列JPA注解来描述实体属性及其与数据库表字段之间的映射关系。以下是几个常用的注解:

  • @Entity:标记一个类为JPA实体。
  • @Table:指定实体对应的数据库表名(可选,默认与类名相同)。
  • @Id:标识主键字段。
  • @GeneratedValue:指定主键生成策略(如自增、UUID等)。
  • @Column:定义列名、长度、是否允许为空等属性。
  • @Temporal:用于日期时间类型的字段,指定存储格式(如TIMESTAMP、DATE等)。

@Entity
@Table(name = "users")
public class User {@Id@GeneratedValue(strategy = GenerationType.IDENTITY)private Long id;@Column(nullable = false, length = 50)private String username;@Column(nullable = false, length = 100)private String email;// Getters and Setters...
}

关联映射

当涉及到多个实体之间的关联时,可以使用以下注解来描述一对多、多对一、多对多等关系:

  • @ManyToOne 和 @OneToMany
  • @ManyToMany
  • @JoinColumn:定义外键约束。
  • @JoinTable:用于多对多关系,指定中间表及其关联字段。
@Entity
public class Post {@Id@GeneratedValue(strategy = GenerationType.IDENTITY)private Long id;@ManyToOne(fetch = FetchType.LAZY)@JoinColumn(name = "user_id", nullable = false)private User author;@OneToMany(mappedBy = "post", cascade = CascadeType.ALL, orphanRemoval = true)private List<Comment> comments = new ArrayList<>();// Getters and Setters...
}@Entity
public class Comment {@Id@GeneratedValue(strategy = GenerationType.IDENTITY)private Long id;@ManyToOne(fetch = FetchType.LAZY)@JoinColumn(name = "post_id", nullable = false)private Post post;private String content;// Getters and Setters...
}

仓库接口

CRUD操作

Spring Data JPA提供了一个名为CrudRepository的基础接口,其中包含了基本的CRUD方法。我们只需要继承这个接口并为每个实体创建相应的仓库接口即可。

public interface UserRepository extends CrudRepository<User, Long> {// Additional custom query methods can be added here.
}

自定义查询方法

除了默认的CRUD方法外,还可以通过命名约定或JPQL/QueryDSL等方式定义自定义查询方法。例如:

public interface UserRepository extends CrudRepository<User, Long> {List<User> findByUsername(String username);@Query("SELECT u FROM User u WHERE u.email LIKE %?1%")List<User> findByEmailLike(String email);
}

事务管理

声明式事务管理

Spring提供了声明式事务管理机制,通过@Transactional注解可以轻松地为服务层方法添加事务支持。如果方法执行期间发生异常,整个事务将会回滚,保证了数据的一致性。

@Service
@Transactional
public class UserService {private final UserRepository userRepository;public UserService(UserRepository userRepository) {this.userRepository = userRepository;}public void createUser(User user) {userRepository.save(user);}
}

编程式事务管理

对于一些复杂的场景,比如需要在一个方法中多次开始和结束事务,编程式事务管理提供了更大的灵活性。

@Autowired
private PlatformTransactionManager transactionManager;public void complexOperation() {DefaultTransactionDefinition def = new DefaultTransactionDefinition();TransactionStatus status = transactionManager.getTransaction(def);try {// Perform operations...transactionManager.commit(status);} catch (Exception e) {transactionManager.rollback(status);throw e;}
}

分页与排序

Spring Data JPA内置了对分页和排序的支持,只需在查询方法签名中添加Pageable参数即可。

public interface UserRepository extends CrudRepository<User, Long> {Page<User> findAll(Pageable pageable);Page<User> findByUsername(String username, Pageable pageable);
}

调用时可以通过PageRequest.of(pageNumber, pageSize)构造分页请求对象,并传入排序规则。

Pageable pageable = PageRequest.of(0, 10, Sort.by(Sort.Direction.ASC, "username"));
Page<User> users = userRepository.findAll(pageable);

性能优化

批量操作

批量插入、更新或删除可以显著提高数据库操作的性能。Hibernate提供了saveOrUpdateAll()、deleteAllInBatch()等方法来支持批量操作。

List<User> users = Arrays.asList(new User("Alice"),new User("Bob"),new User("Charlie")
);
userRepository.saveAll(users);

查询缓存

启用查询缓存可以在一定程度上减少重复查询带来的性能开销。需要注意的是,查询缓存只适用于只读查询,并且可能会增加内存消耗。

@Cacheable(value = "users")
public List<User> findUsersByRole(@Param("role") String role) {return userRepository.findByRole(role);
}

继承策略

JPA支持三种继承映射策略:单表(Single Table)、表每类(Table Per Class)和联合表(Joined Table)。选择合适的策略取决于实体之间的关系和查询模式。

@Entity
@Inheritance(strategy = InheritanceType.SINGLE_TABLE)
@DiscriminatorColumn(name = "user_type")
public abstract class User {// Common fields...
}@Entity
@DiscriminatorValue("admin")
public class Admin extends User {// Admin-specific fields...
}@Entity
@DiscriminatorValue("customer")
public class Customer extends User {// Customer-specific fields...
}

多租户支持

多租户架构允许多个客户共享同一套应用程序实例,但数据彼此隔离。Hibernate提供了两种方式来实现多租户:辨别器(Discriminator)和方案(Schema)。

@Configuration
@EnableTransactionManagement
public class HibernateConfig {@Beanpublic MultiTenantConnectionProvider multiTenantConnectionProvider() {return new MultiTenantConnectionProviderImpl();}@Beanpublic CurrentTenantIdentifierResolver currentTenantIdentifierResolver() {return new CurrentTenantIdentifierResolverImpl();}
}

代码举例:

构建一个简单的博客系统,包含用户、帖子和评论三个实体。首先,按照前面所述步骤创建Spring Boot项目,并定义相应的实体类和仓库接口。然后,编写控制器和服务层代码来处理HTTP请求和业务逻辑。

控制器

@RestController
@RequestMapping("/api/users")
public class UserController {private final UserService userService;public UserController(UserService userService) {this.userService = userService;}@GetMappingpublic ResponseEntity<List<User>> getAllUsers() {return ResponseEntity.ok(userService.getAllUsers());}@PostMappingpublic ResponseEntity<User> createUser(@RequestBody User user) {return ResponseEntity.status(HttpStatus.CREATED).body(userService.createUser(user));}
}

服务层

@Service
@Transactional
public class UserService {private final UserRepository userRepository;public UserService(UserRepository userRepository) {this.userRepository = userRepository;}public List<User> getAllUsers() {return userRepository.findAll();}public User createUser(User user) {return userRepository.save(user);}
}

启动应用程序后,可以通过Postman或浏览器访问API端点进行测试。

最佳实践与注意事项

  • 保持实体类轻量化:避免在实体类中加入过多业务逻辑,尽量将其放在服务层中处理。
  • 合理使用懒加载:对于关联关系,默认采用懒加载方式可以有效减少不必要的查询次数。
  • 注意并发控制:在高并发环境下,考虑使用乐观锁或悲观锁机制来防止数据竞争问题。
  • 充分利用缓存:适当启用查询缓存和二级缓存,可以显著提升读取性能。
  • 定期清理过期数据:对于历史数据较多的应用,建议定期归档或删除不再需要的数据,以维护数据库健康状态。

通过Spring集成JPA/Hibernate进行ORM操作,不仅可以简化数据访问层的实现,还能充分发挥两者的优势,满足复杂的企业级应用需求,欢迎大家一起沟通交流~

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

使用Idea自带的git功能进行分支合并

文章目录 1.背景描述2.分支切换3.分支合并的具体操作4.将在local环境下&#xff0c;从dev合并到qas分支上的代码&#xff0c;推送到远端 1.背景描述 目前在开发的当前项目有四个分支&#xff0c;master(主分支)、pre(预生产分支)、qas(测试分支)、dev(开发分支)&#xff1b; …

EE308FZ_Sixth Assignment_Beta Sprint_Sprint Essay 3

Assignment 6Beta SprintCourseEE308FZ[A] — Software EngineeringClass Link2401_MU_SE_FZURequirementsTeamwork—Beta SprintTeam NameFZUGOObjectiveSprint Essay 3_Day5-Day6 (12.15-12.16)Other Reference1. WeChat Mini Program Design Guide 2. Javascript Style Guid…

凯酷全科技抖音电商服务的卓越践行者

在数字经济蓬勃发展的今天&#xff0c;电子商务已成为企业增长的新引擎。随着短视频平台的崛起&#xff0c;抖音作为全球领先的短视频社交平台&#xff0c;不仅改变了人们的娱乐方式&#xff0c;也为品牌和商家提供了全新的营销渠道。厦门凯酷全科技有限公司&#xff08;以下简…

架构信息收集(小迪网络安全笔记~

附&#xff1a;完整笔记目录~ ps&#xff1a;本人小白&#xff0c;笔记均在个人理解基础上整理&#xff0c;若有错误欢迎指正&#xff01; 2.2 架构信息收集 引子&#xff1a;一个Web应用的构成&#xff0c;由诸多组件&服务相结合&#xff0c;而域名仅是处于Web架构中最表…

一.photoshop导入到spine

这里使用的是 photoshoptospine脚本 下载地址:https://download.csdn.net/download/boyxgb/90156744 脚本的使用,可以通过文件的脚本的浏览,浏览该脚本使用该脚本,也可以将该脚本放在photoshop安装文件夹里的script文件夹下,具体路径:Photoshop\Presets\Scripts,重启photosho…

Mapbox-GL 的源码解读的一般步骤

Mapbox-GL 是一个非常优秀的二三维地理引擎&#xff0c;随着智能驾驶时代的到来&#xff0c;应用也会越来越广泛&#xff0c;关于mapbox-gl和其他地理引擎的详细对比&#xff08;比如CesiumJS&#xff09;&#xff0c;后续有时间会加更。地理首先理解 Mapbox-GL 的源码是一项复…

SparkSQL运行架构及原理

文章目录 SparkSQL运行架构及原理1.1. Catalyst优化器简介1.2. SparkSQL运行架构1.3. SparkSQL解析Core底层原理1.4. 执行计划查看 SparkSQL运行架构及原理 1.1. Catalyst优化器简介 SparkSQL使得我们开发人员可以使用DSL风格的数据来处理数据&#xff0c;甚至可以直接使用SQ…

大数据-254 离线数仓 - Airflow 任务调度 核心交易调度任务集成

点一下关注吧&#xff01;&#xff01;&#xff01;非常感谢&#xff01;&#xff01;持续更新&#xff01;&#xff01;&#xff01; Java篇开始了&#xff01; 目前开始更新 MyBatis&#xff0c;一起深入浅出&#xff01; 目前已经更新到了&#xff1a; Hadoop&#xff0…

昇思25天学习打卡营第33天|共赴算力时代

文章目录 一、平台简介二、深度学习模型2.1 处理数据集2.2 模型训练2.3 加载模型 三、共赴算力时代 一、平台简介 昇思大模型平台&#xff0c;就像是AI学习者和开发者的超级基地&#xff0c;这里不仅提供丰富的项目、模型和大模型体验&#xff0c;还有一大堆经典数据集任你挑。…

Docker 镜像加速和配置的分享 云服务器搭建beef-xss

前言 最近很多的docker镜像加速都鸡鸡了 找点资源是越来越不容易了 什么事docker 因为我是个业余的人 我简单的说 docker就是比如我们的软件商店的 下载 docker镜像&#xff08;之前就是我们在服务器上搭建网站 和环境的很费力费时 之后就有了这个 镜像 &#xff1a;这…

浅谈怎样系统的准备前端面试

前言 创业梦碎&#xff0c;回归现实&#xff0c;7 月底毅然裸辞&#xff0c;苦战两个月&#xff0c;拿到了美团和字节跳动的 offer&#xff0c;这算是从业以来第一次真正意义的面试&#xff0c;遇到蛮多问题&#xff0c;比如一开始具体的面试过程我都不懂&#xff0c;基本一直是…

告别机器人味:如何让ChatGPT写出有灵魂的内容

目录 ChatGPT的一些AI味道小问题 1.提供编辑指南 2.提供样本 3.思维链大纲 4.融入自己的想法 5.去除重复增加多样性 6.删除废话 ChatGPT的一些AI味道小问题 大多数宝子们再使用ChatGPT进行写作时&#xff0c;发现我们的老朋友ChatGPT在各类写作上还有点“机器人味”太重…

【长城杯】Web题 hello_web 解题思路

查看源代码发现路径提示 访问…/tips.php显示无用页面&#xff0c;怀疑…/被过滤&#xff0c;采用…/./形式&#xff0c;看到phpinfo()页面 注意到disable_functions&#xff0c;禁用了很多函数 访问hackme.php,看到页面源码 发现eval函数&#xff0c;包含base64 解密获得php代…

Windows部署Docker及PostgreSQL数据库相关操作

一、Windows安装Docker 1.wsl安装 以管理员身份启动命令行&#xff0c;运行&#xff1a;wsl --install&#xff1b; 安装结束后&#xff0c;重启电脑&#xff0c;以管理员身份启动命令行&#xff0c;运行&#xff1a;wsl --install -d Ubuntu&#xff1b; 中间需要输入用户名…

HTML零基础入门教学

目录 一. HTML语言 二. HTML结构 三. HTML文件基本结构 四. 准备开发环境 五. 快速生成代码框架 六. HTML常见标签 6.1 注释标签 6.2 标题标签&#xff1a;h1-h6 6.3 段落标签&#xff1a;p 6.4 换行标签&#xff1a;br 6.5 格式化标签 6.6 图片标签&a…

Springboot应用开发:工具类整理

目录 一、编写目的 二、映射工具类 2.1 依赖 2.2 代码 三、日期格式 3.1 依赖 3.2 代码 四、加密 4.1 代码 五、Http请求 5.1 依赖 5.2 代码 六、金额 6.1 代码 七、二维码 7.1 依赖 7.2 代码 八、坐标转换 8.1 代码 九、树结构 9.1 代码 9.1.1 节点 9.1…

libaom 源码分析:熵编码模块介绍

AV1 熵编码原理介绍 关于AV1 熵编码原理介绍可以参考:AV1 编码标准熵编码技术概述libaom 熵编码相关源码介绍 函数流程图 核心函数介绍 av1_pack_bitstream 函数:该函数负责将编码后的数据打包成符合 AV1 标准的比特流格式;包括写入序列头 OBU 的函数 av1_write_obu_header…

一个开源的自托管虚拟浏览器项目,支持在安全、私密的环境中使用浏览器

大家好&#xff0c;今天给大家分享一个开源的自托管虚拟浏览器项目Neko&#xff0c;旨在利用 WebRTC 技术在 Docker 容器中运行虚拟浏览器&#xff0c;为用户提供安全、私密且多功能的浏览体验。 项目介绍 Neko利用 WebRTC 技术在 Docker 容器中运行虚拟浏览器&#xff0c;提供…

【已解决】启动此实时调试器时未使用必需的安全权限。要调试该进程,必须以管理员身份运行此实时调试器。是否调试该进程?

【已解决】启动此实时调试器时未使用必需的安全权限。要调试该进程&#xff0c;必须以管理员身份运行此实时调试器。是否调试该进程? 目录一、前言二、具体原因三、解决方法 目录 报错截图 一、前言 进行应用程序开发时&#xff0c;需要对w3wp进行附加调试等场景&#xff…

Docker--Docker Registry(镜像仓库)

什么是Docker Registry&#xff1f; 镜像仓库&#xff08;Docker Registry&#xff09;是Docker生态系统中用于存储、管理和分发Docker镜像的关键组件。 镜像仓库主要负责存储Docker镜像&#xff0c;这些镜像包含了应用程序及其相关的依赖项和配置&#xff0c;是构建和运行Doc…