JPA陷阱/错误

根据我在帮助团队和进行培训方面的经验,这是我遇到的一些陷阱/错误,这些陷阱/错误在使用JPA的基于Java的系统中引起了一些问题。

  • 需要一个公共的无参数构造函数
  • 始终使用双向关联/关系
  • 使用@OneToMany收集可能庞大的集合

需要一个公共的无参数构造函数

是的,JPA @Entity需要零参数(或默认的无参数)构造函数。 但这可以得到protected 您不必将其public 这样就可以实现更好的面向对象的建模,因为您不必强制使用可公开访问的零参数构造函数。

实体类必须具有no-arg构造函数。 实体类也可以具有其他构造函数。 no-arg构造函数必须是public 或protected 。 [强调我的]

–摘自Java Persistence API 2.1规范(Oracle)的2.1

如果要建模的实体在创建时有一些字段需要初始化,则应通过其构造函数来完成。

注意:一些JPA提供程序可以通过在构建时添加一个无参数的构造函数来克服缺失的无参数构造函数。

假设我们正在建模酒店房间预订系统。 在其中,我们可能有诸如房间,预订等之类的实体。预订实体可能需要开始日期和结束日期,因为没有停留时间来创建一个预订实体就没有多大意义。 在保留的构造函数中将开始日期和结束日期作为参数包括在内,将可以提供更好的模型。 保留受保护的零参数构造函数会使JPA满意。

@Entity
public class Reservation { ...public Reservation(RoomType roomType, DateRange startAndEndDates) {if (roomType == null || startAndEndDates == null) {throw new IllegalArgumentException(...);} ...}...protected Reservation() { /* as required by ORM/JPA */ }
}

注意: Hibernate(JPA提供程序)允许将零参数构造函数设为私有。 这使您的JPA代码不可移植到其他JPA提供程序。

它还有助于在零参数构造函数中添加注释,以指示它是出于JPA目的(技术基础结构)而添加的,并且不是域所必需的(业务规则/逻辑)。

尽管我在JPA 2.1规范中找不到它,但可嵌入类也需要一个默认(无参数)构造函数。 就像实体一样,可以将必需的无参数构造函数设为protected

@Embeddable
public class DateRange { ...public DateRange(Date start, Date end) {if (start == null || end == null) {throw new IllegalArgumentException(...);}if (start.after(end)) {throw new IllegalArgumentException(...);} ...}...protected DateRange() { /* as required by ORM/JPA */ }
}

DDD示例项目还通过使它成为包范围来隐藏no-arg构造函数(请参阅Cargo实体类,其中no-arg构造函数位于底部附近)。

始终使用双向关联/关系

JPA上的教学材料通常显示出双向关联。 但这不是必需的。 例如,假设我们有一个包含一个或多个项目的订单实体。

@Entity
public class Order {@Id private Long id;@OneToMany private List<OrderItem> items;...
}@Entity
public class OrderItem {@Id private Long id;@ManyToOne private Order order;...
}

很高兴知道JPA支持双向关联。 但是实际上,这成为维护的噩梦。 如果订单项不必知道其父订单对象,则单向关联就足够了(如下所示)。 ORM只需要知道如何命名多边表中的外键列。 这是通过在关联的一侧添加@JoinColumn批注来提供的。

@Entity
public class Order {@Id Long id;@OneToMany@JoinColumn(name="order_id", ...)private List<OrderItem> items;...
}@Entity
public class OrderItem {@Id private Long id;// @ManyToOne private Order order;...
}

由于OrderItem不再需要保留对Order实体的引用,因此使其变得单向变得更容易。

请注意,有时可能需要双向关联。 实际上,这种情况很少见。

这是另一个例子。 假设您有几个引用某个国家/地区实体的实体(例如,人的出生地,邮政地址等)。 显然,这些实体将引用国家实体。 但是,国家是否必须引用所有这些不同的实体? 很有可能,不是。

@Entity
public class Person {@Id Long id;@ManyToOne private Country countryOfBirth;...
}@Entity
public class PostalAddress {@Id private Long id;@ManyToOne private Country country;...
}@Entity
public class Country {@Id ...;// @OneToMany private List<Person> persons;// @OneToMany private List<PostalAddress> addresses;
}

所以,仅仅因为JPA支持双向关联, 并不意味着你必须!

使用

假设您正在建模银行帐户及其交易。 随着时间的流逝,一个帐户可以进行数千(甚至数百万)笔交易。

@Entity
public class Account {@Id Long id;@OneToMany@JoinColumn(name="account_id", ...)private List<AccountTransaction> transactions;...
}@Entity
public class AccountTransaction {@Id Long id;...
}

对于只有少量交易的帐户,似乎没有任何问题。 但是随着时间的流逝,当一个帐户包含成千上万个(如果不是上百万个)交易时,您很可能会遇到内存不足的错误。 那么,有什么更好的映射方法呢?

如果不能确保关联的多面中的最大元素数都可以全部加载到内存中,则最好在@ManyToOne的另一侧使用@ManyToOne

@Entity
public class Account {@Id Long id;// @OneToMany private List<AccountTransaction> transactions;...
}@Entity
public class AccountTransaction {@Id Long id;@ManyToOneprivate Account account;...public AccountTransaction(Account account, ...) {...}protected AccountTransaction() { /* as required by ORM/JPA */ }
}

要检索一个帐户可能数千(如果不是几百万)的交易,请使用支持分页的存储库。

@Transactional
public interface AccountTransactionRepository {Page<AccountTransaction> findByAccount(Long accountId, int offset, int pageSize);...
}

要支持分页,请使用Query对象的setFirstResult(int)setMaxResults(int)方法。

摘要

我希望这些说明可以帮助开发人员避免犯这些错误。 总结一下:

  • 需要公众。 JPA要求的无参数构造函数可以设为publicprotected 。 如果需要,可以考虑对其进行protected
  • 一直使用 考虑单向而不是双向关联/关系。
  • 使用 避免使用@OneToMany收集可能庞大的集合。 考虑@ManyToOne映射关联/关系的@ManyToOne端,并支持分页。

翻译自: https://www.javacodegeeks.com/2016/02/jpa-pitfalls-mistakes.html

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

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

相关文章

CSGL

glShadeModel void glShadeModel(GLenum mode) GL_FLAT/【GL_SMOOTH】 着色技术选择 glClearDepth GL.glClearDepth(depth); glClearDepth&#xff1a;设置深度缓存的清除值 参数 depth 指定清除深度缓存时使用的深度值。 说明 本函数指定用glClear清除深度缓存时所使用的深度值…

强大的Canvas开源库Fabric.js简介与开发指南

什么是Fabric.js&#xff1f;Fabric.js 是一个强大且简单的Javascript HTML5 Canvas库。官网地址&#xff1a;http://fabricjs.com/为什么要使用Fabric.js&#xff1f;Canvas提供一个好的画布能力, 但是Api不够友好。绘制简单图形其实还可以, 不过做一些复杂的图形绘制, 编写一…

模拟qq斗地主-准备发牌抢地主都是农民下一轮准备

为什么要搞这样一个项目&#xff1f;&#xff1f; 1&#xff0c;满足自己的java网络多线程编程的欲望&#xff01;因为之前一直都是搞web开发&#xff0c;服务器和客户端数据交流人家web服务器早就给你搞好了&#xff0c;比如tomcat,jetty...等等&#xff0c;其实之前脑子里就有…

rube3xxx_Rube GoldbergSpring整合

rube3xxxSpring Integration为集成系统所涉及的一些复杂性提供了非常好的抽象-Spring Integration从Integration的角度完美地满足了Facade的定义-简化了对复杂基础系统的访问。 为了说明这一点&#xff0c;请考虑一个简单的系统&#xff0c;该系统仅接收一条消息&#xff0c;然…

纯CSS实现React Logo图形,内含详细解析

以上是将要实现的效果&#xff0c;Javascript框架React的Logo图形&#xff0c;首先我们来拆解下&#xff0c;它包括三个交叉的椭圆和中间一个圆点&#xff0c;所以我们Html元素可以用以下代码实现&#xff1a;<div class"main"><div class"ellipse ell…

学生ID查询

var http require("http");var server http.createServer(function(req,res){//得到urlvar userurl req.url;res.writeHead(200,{"Content-Type":"text/html;charsetUTF8"})//substr函数来判断此时的开头if(userurl.substr(0,9) "/stud…

平等还是认同?

将对象存储在集合中时&#xff0c;同一对象永远不能添加两次非常重要。 这是集合的核心定义。 在Java中&#xff0c;使用两种方法来确定两个引用的对象是否相同或它们是否可以同时存在于同一Set中。 equals&#xff08;&#xff09;和hashCode&#xff08;&#xff09;。 在本文…

二次优化大招(由泰勒公式推出最值条件)

经过前两篇的铺垫&#xff0c;这一篇迎来了高潮。先说一句&#xff1a;特征值大法好&#xff01; 在开始正文之前&#xff0c;先看以下简单的推导 有了这些&#xff0c;理解下面的泰勒公式推出最值条件就容易了 转载于:https://www.cnblogs.com/Mr-ZeroW/p/7764916.html

前端自动化测试浅析

前言&#xff1a;测试简介前端常见的问题&#xff1a;修改某个模块功能时&#xff0c;其它模块也受影响&#xff0c;很难快速定位bug多人开发代码越来越难以维护不方便迭代&#xff0c;代码无法重构代码质量差增加自动化测试后&#xff1a;我们为核心功能编写测试后可以保障项目…

使用SpringData出现java.lang.AbstractMethodError

最近学习一下SpringData&#xff0c;在添加SpringData支持的时候&#xff0c;出现了这样的问题&#xff1a; SpringData需要的jar有:spring-data-jpa.jar spring-data-commons.jar slf4j-api.jar 没有添加slf4j也会出现一个异常&#xff0c;不过那个异常说的非常明确&#xf…

2021这份电子书单请收好(品类齐全)!

2021年已经快过去一个月了&#xff0c;今年的读书计划有没有安排上&#xff0c;这里有份长长的书单&#xff0c;多年的积累&#xff0c;品类齐全&#xff0c;赶快从中挑几本加入今年的读书计划里。•《货币战争2&#xff1a;金权天下》 - 宋鸿兵.mobi•《圣经》中英对照豪华版 …

在Kotlin中使用libGDX

最近&#xff0c;我一直在阅读有关不同语言的信息&#xff0c;以及它们可以为已经拥挤的软件开发人员带来什么&#xff0c;而一种语言对我来说很突出&#xff1a;Kotlin。 &#xff08; https://kotlinlang.org/ &#xff09; 这是一种相对较新的语言&#xff08;成立于2011年…

练习1-2:编写一个 JAVA 程序,实现输出考试成绩的前三名。

1、 考试成绩已保存在数组 scores中&#xff0c;数组元素依次为 89 , -23 , 64 , 91 , 119 , 52 , 732、 要求通过自定义方法来实现成绩排名并输出操作&#xff0c;将成绩数组作为参数传入3、 要求判断成绩的有效性&#xff08; 0—100 &#xff09;&#xff0c;如果成绩无效&a…

【重学JS系列】slice用法大合集

让我们回顾下slice的日常用法slice 工作原理在深入研究一些更高级的用法之前&#xff0c;让我们看一下slice方法的基础知识。如MDN文档&#xff0c;slice 是数组上的一个方法&#xff0c;它最多有两个参数:arr.slice([begin[, end]])begin从该索引处开始提取原数组中的元素,如果…

redux-4-ways

https://medium.com/react-native-training/redux-4-ways-95a130da0cdc转载于:https://www.cnblogs.com/skating/p/7774090.html

【Github开源】一站搞定各种开发文档

开发者的苦恼&#xff1a;经常要在多个API文档中切换&#xff0c;浏览器书签栏收藏各种语言相关的接口说明文档。无意中在Github上发现DevDocs[1]这个开源项目&#xff0c;它是一个把所有开发相关的文档以web的形式做了一个综合的网站&#xff0c;并提供搜索&#xff0c;离线访…

javafx 表单_JavaFX 2:创建登录表单

javafx 表单在本教程中&#xff0c;我将使用JavaFX 2和CSS设计一个外观漂亮的Login Form 。 它是经典的登录表单&#xff0c;带有用户名和密码以及登录按钮。 为了遵循本教程&#xff0c;我强烈建议您查看以下这些教程&#xff1a; Eclipse IDE中的JavaFX 2入门 JavaFX 2&…

中高级JavaScript易错面试题

写出下题的输出 1、函数的实参与形参length var length 10; function fn() {console.log(this.length); } var obj {length: 5,method: function(fn) {fn();arguments[0]();} }; console.log(obj.method(fn, 1));  // 0 2 我们都知道&#xff0c;[1, 2, 3].length可以得到3…

静音设计模式

您最近是否遵循Mute-Design-Pattern™编写了大量代码&#xff1f; 例如 try {complex();logic();here(); } catch (Exception ignore) {// Will never happen heheSystem.exit(-1); }Java 8有一个更简单的方法&#xff01; 只需将这个非常有用的工具添加到您的Utilities或Hel…

【详细教程】教你如何使用Node + Express + Typescript开发一个应用

Express是nodejs开发中普遍使用的一个框架&#xff0c;下面要谈的是如何结合Typescript去使用。 目标 我们的目标是能够使用Typescript快速开发我们的应用程序&#xff0c;而最终我们的应用程序却是编译为原始的JavaScript代码&#xff0c;以由nodejs运行时来执行。 初始化设置…