JPA和Hibernate的相关使用技巧
介绍
尽管有SQL标准,但每个关系数据库终将是唯一的,因此你需要调整数据访问层,以便充分利用在使用中的关系数据库。
在本文中,我们将介绍在使用带有JPA和Hibernate的MySQL时,为了提高性能,我们可以做哪些事情。
不要使用AUTO标识符GeneratorType
每个实体都需要标识符,标识符惟一地标识与该实体关联的表记录。JPA和Hibernate允许根据三种不同的策略自动生成实体标识符:
- IDENTITY
- SEQUENCE
- TABLE
正如我在这篇文章中所解释的,当增加数据库连接数时,TABLE标识符策略不会缩放。而且,即使是一个数据库连接,标识符生成响应时间比使用IDENTITY或SEQUENCE大十倍。
如果你使用AUTO GenerationType
:
@Id
@GeneratedValue(strategy = GenerationType.AUTO)
private Long id;
Hibernate 5将会退回到使用TABLE生成器,这对性能不利。
正如我在这篇文章中所解释的,你可以使用以下映射轻松解决此问题:
@Id
@GeneratedValue(strategy= GenerationType.AUTO, generator="native")
@GenericGenerator(name = "native", strategy = "native")
private Long id;
本地生成器将选择IDENTITY而不是TABLE。
IDENTITY生成器禁用JDBC批处理插入
MySQL 5.7和8.0都不支持SEQUENCE对象,因此你需要使用IDENTITY。但是,正如我在这篇文章中所解释的,IDENTITY生成器可以防止Hibernate使用JDBC批量插入。
JDBC批量更新和删除不受影响,只有INSERT语句不能被批处理,因为在Persistence Context被刷新之前,INSERT语句已被执行,从而Hibernate知道要分配给持久化实体什么实体标识符。
如果要解决此问题,则必须通过不同的框架,如jOOQ,执行JDBC批处理插入。
通过Docker和tmpfs加速集成测试
MySQL和MariaDB在不得不丢弃数据库模式的时候,以及每次新的集成测试即将运行因而重新创建它的时候,是非常慢的。但是,你可以在Docker和tmpfs的帮助下轻松解决此问题。
正如我在这篇文章中所解释的,通过映射内存中的数据文件夹,集成测试的运行速度将与有内存数据库(如H2或HSQLDB)时的速度相同。
对非结构化数据使用JSON
即使是在你使用RDBMS的时候,肯定也有很多次想要存储非结构化数据:
- 来自客户端,如JSON的数据,需要被解析并插入到我们的系统中。
- 可以缓存的图像处理结果以保存再处理
虽然本机不支持,但是你可以轻松地将Java对象映射到JSON列。甚至可以将JSON列类型映射到Jackson JsonNode。
更重要的是,你甚至不必编写这些自定义类型,可以从Maven Central中抓取:
<dependency>
<groupId>com.vladmihalcea</groupId>
<artifactId>hibernate-types-52</artifactId>
<version>1.0.0</version>
</dependency>
很酷,对吧?
使用存储过程来保存数据库
在处理大量数据时,将所有数据移入和移出数据库并不是非常高效。不过,通过调用存储过程对数据库端进行处理会好很多。
小心ResultSet流
SQL流在两层应用程序中是很有意义的。如果你要执行ResultSet流,那么你也得注意JDBC驱动程序。在MySQL上,你需要将Statement大小设置为Integer.MIN_VALUE。
然而,对于基于Web的应用程序,分页更为合适。JPA 2.2甚至引入了对Java 1.8 Stream方法的支持,但执行计划可能不如使用SQL级别分页时那么高效。
PreparedStatements可能会被仿真
你可能以为,既然Hibernate默认使用PreparedStatements
,那么所有语句都是像这样执行的:
实际上,更像是这样执行的:
正如我在这篇文章中所解释的,除非你设置了useServerPrepStmts
MySQL JDBC驱动程序属性,否则PreparedStatements将在JDBC驱动程序级别进行仿真以保存一个额外的数据库。
始终结束数据库事务
在关系数据库中,每个语句都在给定的数据库事务中执行。因此,事务是不可选的。
但是,你应该始终通过提交或回滚来结束当前正在运行的事务。忘记结束事务可能会导致持续被锁很长时间,同时也会阻止MVCC清理过程回收不再需要的旧元组或索引条目。
递交日期/时间没有那么容易
编程中有两件非常复杂的事情:
- 处理编码
- 处理跨多个时区的日期/时间
为了解决第二个问题,最好在UTC时区中保存所有时间戳。但是,当使用MySQL时,你还需要将useLegacyDatetimeCode
JDBC Driver配置属性设置为false
。
结论
正如你所看到的,在使用带有JPA和Hibernate的MySQL时,要记住许多事情。因为MySQL是最为广泛部署的RDBMS之一,并被绝大多数的Web应用程序所使用,所以,了解所有这些技巧并调整数据访问层来最大限度地利用它非常有用。
JPA和Hibernate的相关使用技巧的更多相关文章
- 三年总结出来的11个JPA和Hibernate查询配置小技巧
JPA和Hibernate提供了一系列暗示hints能够帮助你更好地定制你的查询语言,这些小暗示或暗语是一种附加信息,你可以利用这些暗语做很多事情,比如设置查询的timeout,使用实体图或定义查询缓 ...
- JPA入门例子(采用JPA的hibernate实现版本) 转
JPA入门例子(采用JPA的hibernate实现版本) jpahibernate数据库jdbcjava框架(1).JPA介绍: JPA全称为Java Persistence API ,Java持久化 ...
- JPA(Hibernate)
JPA 1,JPA:Java Persistence API.JPA通过JDK 5.0注解-关系表的映射关系,并将运行期的实体对象持久化到数据库中.JPA是JavaEE中的标准.JPA标准只提供了一套 ...
- JPA和Hibernate到底是什么关系???
转自:https://www.cnblogs.com/mosoner/p/9494250.html 在学习框架的过程中,发现学的东西很多,但是感觉他们之间的联系区别都不是很了解,知道JPA可以去实现持 ...
- 简述 JPA 与 Spring Data JPA 与 Hibernate
1.JPA是什么?以及相关概述 JPA的是 Java Persistence API 的简写,是Sun官方提出的一种ORM规范! Sun提出此规范有2个原因: 1.简化现有Java EE和Java S ...
- JPA 入门程序及相关注解
1. 概述 JPA(Java Persistence API):用于对象持久化的API; JPA本质上是一种ORM规范,不是ORM框架;提供了一些编程的API接口; Hibernate是实现; 1.1 ...
- javaweb各种框架组合案例(六):springboot+spring data jpa(hibernate)+restful
一.介绍 1.springboot是spring项目的总结+整合 当我们搭smm,ssh,ssjdbc等组合框架时,各种配置不胜其烦,不仅是配置问题,在添加各种依赖时也是让人头疼,关键有些jar包之间 ...
- JPA入门例子(采用JPA的hibernate实现版本) --- 会伴随 配置文件:persistence.xml
JPA入门例子(采用JPA的hibernate实现版本) 分类: j2se2011-03-30 16:09 45838人阅读 评论(9) 收藏 举报 jpahibernate数据库jdbcjava框架 ...
- JPA和hibernate的关系
实际上,JPA的标准的定制是hibernate作者参与定制的,所以JPA是hibernate的一个总成,可以这么理解
随机推荐
- Java类的初始化与实例对象的初始化
Java对象初始化详解 2013/04/10 · 开发 · 1 评论· java 分享到:43 与<YII框架>不得不说的故事—扩展篇 sass进阶篇 Spring事务管理 Android ...
- Ceph的工作原理及流程
本文将对Ceph的工作原理和若干关键工作流程进行扼要介绍.如前所述,由于Ceph的功能实现本质上依托于RADOS,因而,此处的介绍事实上也是针对RADOS进行.对于上层的部分,特别是RADOS GW和 ...
- Docker垃圾回收机制
由Docker垃圾回收机制引发的一场血案 AlstonWilliams 关注 2017.04.01 19:00* 字数 1398 阅读 253评论 0喜欢 0 今天早晨,在我还没睡醒的时候,我们团队中 ...
- 我的Linux之路——xshell连接linux虚拟机
出自:https://www.linuxidc.com/Linux/2016-08/134087.htm xshell 5登录本地虚拟机的具体操作步骤如下: 1.首先打开虚拟机,登录到操作系统; 2. ...
- JAVA压缩 解压缩zip 并解决linux下中文乱码
1. [代码][Java]代码 1:再压缩前,要设置linux模式, 需要使用第三方ant-1.6.5.jar 如果是文件目录,则ZipEntry zipEntry=new ZipEntry(b ...
- think in avalon
1.不要设计,也不要通过DOM操作去改变你的页面 你用jQuery去设计一个页面,并让它动起来.这是因为jQuery就是让一切简单的事情变复杂的罪魁祸首. 但是用avalon,你必须从零开始去构思你的 ...
- Mysql 中的伪列用法1
SELECT ( @rowNO := @rowNo + 1 ) AS rowno, A.*FROM ( SELECT * FROM t_user ) a, ( SELECT @rowNO := 0 ) ...
- 产品负责人(Product Owner)的主要职责和技能
角色介绍 产品负责人以下简称PO,他是有授权的产品领导力核心,组成Scrum团队三个角色之一. PO担任的是产品经理的角色. PO的主要职责 1.对产品的ROI负责. ROI = profitabil ...
- flex 设置换行flex-wrap
flex 设置flex-wrap 换行 本来预想的正常情况下,代码应该如下: ul { width: 100%; display: flex; flex-wrap: wrap; li { ; widt ...
- 使用tcmalloc编译出现undefined reference to `sem_init'
tcmalloc是Google开源的一个内存管理库, 作为glibc malloc的替代品,效率大概是gclibc malloc的几倍.想在工程中用上tcmalloc非常的简单,我们采用了静态编译的方 ...