了解如何使用Hibernate轻松解决最常见的问题

Hibernate可能是市场上最受欢迎的JPA实现,您可以在许多地方看到它,例如:

  • 您自己使用过的项目数,
  • 需要Hibernate经验的职位数量,当然还有
  • 互联网上发布的问题和例外数量

在Takipi,重点是查找和修复异常。 因此,我将重点关注列表中的最后一点,并与您分享我可能已经解决,解释,写博客和抱怨的5个Hibernate异常,这是我与Hibernate合作超过15年以来最多的。

尽管他们并没有为十大例外类型提供支持 ,但谷歌的快速搜索告诉我,我并不是唯一面对这些问题的人。

但是在我们探讨不同的例外之前,这篇文章是一篇很长的文章,我在免费备忘单中总结了最重要的观点。 您可以在这篇文章的末尾下载它。

1. LazyInitializationException

如果您尝试在没有活动会话的情况下尝试访问另一个实体的未初始化关系,则Hibernate会抛出LazyInitializationException。 您可以在以下代码片段中看到一个简单的示例。

EntityManager em = emf.createEntityManager();
em.getTransaction().begin();Author a = em.find(Author.class, 1L);em.getTransaction().commit();
em.close();log.info(a.getFirstName() + " " + a.getLastName() + " wrote "+a.getBooks().size() + " books.");

好的,您现在可能会说您永远不会那样做。 尽管您可能永远不会在应用程序中使用完全相同的代码是正确的,但您可以无意间轻松地执行相同的操作。

最受欢迎的方法是在您的业务层中未初始化的演示层中访问与FetchType.LAZY的关系。 您可以在受欢迎的论坛中找到很多此类问题,并提出了许多不良的解决方法。

请不要在视图反模式中使用打开的会话。 它造成的危害更大,然后才带来收益。

修复LazyInitializationException的最佳方法是在业务层中初始化所需的关系。 但是不要仅仅因为可能有一个客户需要其中之一就初始化所有关系。 出于性能原因,您应该只初始化所需的关系。

JPA和Hibernate提供了不同的选项来初始化延迟获取的关系 。 我个人最喜欢的是@NamedEntityGraph ,它提供了独立于查询的方式来定义将随查询获取的实体图。

您可以在以下代码段中看到一个简单图形的示例。 它获取一个Author实体的Book关系。

@NamedEntityGraph(name = "graph.AuthorBooks", attributeNodes = @NamedAttributeNode("books"))

您可以在Hibernate可用的任何文件中定义@NamedEntityGraph。 我更喜欢在打算与之一起使用的实体上进行操作。

如您所见,定义图形不需要做太多的事情。 您只需要提供名称和@NamedAttributeNode批注的数组即可定义Hibernate从数据库中获取的属性。 在此示例中,只有book属性将关系映射到Book实体。

然后,您可以提供此图作为Hibernate的提示,以定义应使用给定查询初始化的关系。 您可以在以下代码段中看到一个示例。

EntityManager em = emf.createEntityManager();
em.getTransaction().begin();EntityGraph<?> graph = em.getEntityGraph("graph.AuthorBooks");
HashMap<String, Object> properties = new HashMap<>();
properties.put("javax.persistence.fetchgraph", graph);Author a = em.find(Author.class, 1L, properties);em.getTransaction().commit();
em.close();log.info(a.getFirstName() + " " + a.getLastName() + " wrote "+a.getBooks().size() + " books.");

如您所见,我首先在EntityManager上调用getEntityGraph(String name)方法以获取实体图的实例。 在下一步中,我将创建带有查询提示的HashMap并将该图添加为javax.persistence.fetchgraph。

在最后一步中,我将查询提示作为find方法的附加参数。 这告诉Hibernate初始化与Book实体的关系,并且我可以在没有活动的Hibernate会话的情况下调用getBooks()方法。

2. OptimisticLockException

另一个非常常见的异常是OptimisticLockException。 当您使用乐观锁定并检测到实体的更新冲突时,Hibernate会抛出该错误。 发生这种情况最常见的原因有两个:

  1. 2个用户尝试在几乎相同的时间点更新同一实体。
  2. 1个用户对同一实体执行了2次更新,并且您没有刷新客户端中的实体表示,因此第一次更新后版本值未更新。

在下面的代码片段中,您可以看到具有2个并发更新的测试用例。

// EntityManager and transaction 1
EntityManager em = emf.createEntityManager();
em.getTransaction().begin();// EntityManager and transaction 2
EntityManager em2 = emf.createEntityManager();
em2.getTransaction().begin();// update 1
Author a = em.find(Author.class, 1L);
a.setFirstName("changed");// update 2
Author a2 = em2.find(Author.class, 1L);
a2.setFirstName("changed");// commit transaction 1
em.getTransaction().commit();
em.close();// commit transaction 2
try {em2.getTransaction().commit();Assert.fail();} catch (RollbackException e) {Assert.assertTrue(e.getCause() instanceof OptimisticLockException);log.info("2nd transaction failed with an OptimisticLockException");}em2.close();

如您所见,我使用两个独立的EntityManager,并使用它们两个启动事务,获取ID为1的Author实体,并更新名字属性。

在我尝试提交第二个事务并为该Author实体的并发更新进行Hibernate检查之前,该方法可以正常工作。 当然,在实际应用中,这将通过两次并行调用同一方法来完成。

如果使用Takipi ,则可以在发生异常时查看所有变量的状态,这对于识别第二个更新调用的来源很有用。

Takipi的错误分析屏幕

在不引入悲观锁定的情况下,您无法做很多事情来避免这种异常,这会牺牲您的应用程序的性能。 只需尝试尽可能频繁地更新客户端中的实体表示,并保持更新操作越短越好。 那应该避免大多数不必要的OptimisticLockException,并且您需要在客户端应用程序中处理其余的剩余部分。

但是,如果只有一个用户自己导致OptimisticLockException,那么您会发现一个可以轻松修复的错误。 如果您使用乐观锁定,则Hibernate将使用version列来跟踪实体的当前版本并防止并发修改。 因此,您需要确保在用户触发实体上的任何更改后,客户端始终更新其对实体的表示。 而且,您的客户端应用程序也不应缓存实体或代表它的任何值对象。

3. org.hibernate.AnnotationException:未知的Id.generator

这是由错误的实体映射引起的,在开发过程中可能会遇到它。 原因很简单,您可以在@GeneratedValue批注中引用未知的序列生成器,如下面的代码片段所示。

@Id
@GeneratedValue(strategy = GenerationType.SEQUENCE, generator = "authorSequence")
@Column(name = "id", updatable = false, nullable = false)
private Long id;

@GeneratedValue批注允许您定义主键值的生成策略。 在前面的代码片段中,我想使用数据库序列,并提供“ authorSequence”作为生成器的名称。

现在,许多开发人员期望“ authorSequence”将成为Hibernate将使用的数据库序列的名称。 事实并非如此。 这是@SequenceGenerator的名称,可用于提供有关Hibernate将使用的数据库序列的更多信息。

但是@SequenceGenerator的定义丢失了,因此Hibernate引发了AnnotationException。 要解决此问题,您必须像在以下代码片段中一样添加一个@SequenceGenerator批注。

@Id
@GeneratedValue(strategy = GenerationType.SEQUENCE, generator = "authorSequence")
@SequenceGenerator(name = "authorSequence", sequenceName = "author_seq", initialValue = 1000)
@Column(name = "id", updatable = false, nullable = false)
private Long id;

@SequenceGenerator批注允许您提供有关数据库序列以及Hibernate如何使用它的更多信息。 在此代码段中,我设置了序列的名称,即“ author_seq”,并将其初始值设置为1000。

您还可以指定序列所属的数据库模式以及Hibernate可以用于性能优化的分配大小。 您可以在以下文章中了解有关ID生成器的更多信息 。

4. QuerySyntaxException:表未映射

这是另一个典型的映射错误。 在大多数项目中,数据库架构已经存在或独立于您的实体映射定义。 那是一件好事。 请正确设计数据库模式,不要让Hibernate为您生成它!

如果您希望Hibernate在启动时设置数据库,最好提供一个SQL脚本,而不是让Hibernate根据您的实体映射生成数据库架构。

现在,回到QuerySyntaxException。 如果数据库模式是独立于您的实体定义的,则通常会遇到以下情况:默认表名与现有表的名称不匹配,或者该表是其他数据库模式的一部分。

在这种情况下,可以为架构和表名提供@Table批注,如以下代码片段所示。

@Entity
@Table(name = "author", schema = "bookstore")
public class Author implements Serializable {…
}

5. org.hibernate.PersistentObjectException:分离的实体传递给持久化

此列表中的最后一个异常可能有多种原因,并且都是错误:

  1. 您尝试保留一个新实体并提供主键值,但是实体映射定义了一种生成它的策略。
  2. 您尝试保留一个新实体,并且持久性上下文已经包含具有给定ID的实体。
  3. 您尝试保留一个分离的实体而不是合并它。

第一个很容易修复,不提供主键值或删除主键生成策略。

仅当您自己管理主键值并且您的算法创建重复项时,才会发生第二种情况。 解决此问题的首选方法是让Hibernate使用数据库序列生成主键值,而不是实现自己的算法。

这并非总是可能的,在这种情况下,您必须测试和调试用于生成主键值的算法。 根据算法的不同,这可能是一项繁琐且耗时的任务。

第三种经常发生在您在客户端中使用实体时,客户端调用了错误的服务器方法,该方法将保留新实体而不是更新现有实体。 解决此错误的明显方法是修复客户端中的呼叫。

另外,您可以在服务器端执行某些操作来避免此类问题,例如使用特定的值对象创建用例,而不是在同一服务器方法中处理创建和更新用例。 这使客户开发人员更容易找到并调用正确的方法,并避免了此类问题。

摘要和备忘单

这些是我最常遇到的5个Hibernate Exception,以及如何修复它们。 如您所见,异常及其原因非常不同。 其中一些仅在开发期间发生,而另一些会在生产中对您造成打击。 因此,最好提防并确保您熟悉这些问题。 为了让您更轻松,我准备了一份备忘单,解释了本文中提到的5个例外 。

翻译自: https://www.javacodegeeks.com/2016/06/5-common-hibernate-exceptions-fix.html

5个常见的Hibernate异常及其解决方法相关推荐

  1. Maven常见异常及解决方法

    异常1: [ERROR] Failed to execute goal on project biz_zhuhai: Could not resolve dependencies for projec ...

  2. mysql 死锁原因_Mysql并发时经典常见的死锁原因及解决方法

    1.mysql都有什么锁 MySQL有三种锁的级别:页级.表级.行级. 表级锁:开销小,加锁快:不会出现死锁:锁定粒度大,发生锁冲突的概率最高,并发度最低. 行级锁:开销大,加锁慢:会出现死锁:锁定粒 ...

  3. 松下a6伺服驱动连接光栅尺_FANUC常见伺服报警及故障解决方法

    FANUC常见伺服报警及故障解决方法 FANUC常见伺服报警及故障解决方法 偶尔SV0435:逆变器DC链路电压低报警 1.确认DCLINK母线接线端子螺丝是否锁紧; 2.如果发生全轴或多轴报警时,请 ...

  4. 常见的 OOM 原因及其解决方法(OutOfMemoryError)

    当 JVM 内存严重不足时,就会抛出 java.lang.OutOfMemoryError 错误.本文总结了常见的 OOM 原因及其解决方法,如下图所示.如有遗漏或错误,欢迎补充指正. 1.Java ...

  5. java oom_Java中常见OOM的场景及解决方法

    OOM for Heap  (java.lang.OutOfMemoryError: Java heap space) 分析 此OOM是由于JVM中heap的最大值不满足需要,将设置heap的最大值调 ...

  6. 线程访问 DevExpress控件异常时 解决方法

    线程访问 DevExpress控件异常时 解决方法 Control.CheckForIllegalCrossThreadCalls = false; DevExpress.Data.CurrencyD ...

  7. 开机启动失败_电脑常见开机引导错误的解决方法

    电脑在开机启动过程中,经常会出现开机失败的故障,开机失败比较常出现在开机引导项错误的问题上,比如引导文件没了,硬盘有问题等等,有些小问题重启几遍可以解决,有些问题就只能修复.这边小编跟大家整理分享几个 ...

  8. 计算机错误符号,解析Excel中常见的错误符号以及解决方法

    解析Excel中常见的错误符号以及解决方法分享给大家, Excel 电子表格是很多人都要使用的软件,也相信很多人都会用,但是用得好不好就差别很大了,用得好的话可以让工作效率大大提高,但关于Excel的 ...

  9. JQuery中使用Ajax赋值给全局变量失败异常的解决方法,jqueryajax

    我们在用JQuery的Ajax从后台提取数据后想把它赋值给全局变量,但是却怎么都赋不进,为什么呢? 原因其实很简单,我们用的Ajax是异步操作,也就是说在你赋值的时候数据还没提取出来,你当然赋不进去, ...

最新文章

  1. WMI技术介绍和应用——查询磁盘分区和逻辑磁盘信息
  2. 最全芯片产业报告出炉,计算、存储、模拟IC一文扫尽
  3. Python学习【第6篇】:Python之文件操作
  4. python中怎么比较两个列表-python中比较两个列表的实例方法
  5. 全球及中国水彩调色板行业销售前景与投资商机研究报告2022版
  6. Spring构建微服务
  7. 玩人工智能的你必须知道的语音识别技术原理
  8. 数据表 高水位 mysql_Oracle中的高水位(HWM)
  9. 台州银行登录显示服务器异常,手把手教你设置台州银行网上银行【处理办法】...
  10. pytorch 支持amd显卡吗_AMD平台上配置PyTorch+Apex开发环境
  11. 受汉城改名鼓舞,世界各地掀起改名热潮,尤其是那个小日本(笑话)
  12. 使用Timer实现Flutter启动页
  13. 企业微信员工能私加客户吗?员工私自联系客户企业是否知道?
  14. ATM机程序Linux,c语言模拟银行ATM机程序
  15. CentOS Stream8安装oh my zsh
  16. 前端开发相关的学习网站
  17. 图像增强及直方图均衡化在图像去雾上的应用(附matlab代码)
  18. 国内网友设计诺基亚Lumia 930概念机
  19. Python BeautifulSoup
  20. 翻译|《Word Power Made Easy》(vii~xii)

热门文章

  1. 消息中间件ActiveMQ、RabbitMQ、RocketMQ、ZeroMQ、Kafka如何选型
  2. 接口 DataInput
  3. (十三)RabbitMQ使用详解
  4. 【Mysql】win10上Mysq的l安装
  5. 利用bladex+avue实现一对多的关系
  6. 利用老毛头启动盘重装win7
  7. java实现人脸识别源码【含测试效果图】——DaoImpl层(BaseDaoUtilImpl)
  8. 第7步 mybatis-generator dao层生成器
  9. 常用的数据交换格式有哪些_高程数据格式介绍
  10. jvm(11)-晚期(运行期)优化