Java:在JPA中使用规范模式

本文是在Java中使用规范模式的简介。 我们还将看到如何将经典规范与JPA Criteria查询结合使用,以从关系数据库中检索对象。

在本文中,我们将使用以下Poll类作为创建规范的示例实体。 它表示具有开始和结束日期的民意调查。 在这两个日期之间的时间中,用户可以在不同的选择之间进行投票。 到达结束日期之前,管理员还可以锁定投票。 在这种情况下,将设置锁定日期。

@Entity
public class Poll { @Id@GeneratedValueprivate long id;private DateTime startDate; private DateTime endDate;private DateTime lockDate;@OneToMany(cascade = CascadeType.ALL)private List<Vote> votes = new ArrayList<>();}

为了提高可读性,我跳过了用于映射此示例中不需要的Joda DateTime实例和字段的getter,setter,JPA批注(例如民意调查中提出的问题)。

现在假设我们要实现两个约束:

  • 如果未锁定且startDate <now <endDate,则轮询当前正在运行
  • 如果民意调查的投票数超过100并且未锁定,则该投票很受欢迎

我们可以从向Poll添加适当的方法开始,例如:poll.isCurrentlyRunning()。 另外,我们可以使用类似pollService.isCurrentlyRunning(poll)的服务方法。 但是,我们还希望能够查询数据库以获取所有当前正在运行的民意调查。 因此,我们可以添加DAO或存储库方法,例如pollRepository.findAllCurrentlyRunningPolls()。

如果采用这种方式,我们将在两个不同的位置两次实现isCurrentlyRunning约束。 如果我们要结合约束,事情就会变得更糟。 如果我们想查询数据库以获取当前正在运行的所有流行民意测验的列表怎么办?

这是规范模式派上用场的地方。 使用规范模式时,我们将业务规则移到称为规范的额外类中。

要开始使用规范,我们创建一个简单的界面和一个抽象类:

public interface Specification<T> {  boolean isSatisfiedBy(T t);  Predicate toPredicate(Root<T> root, CriteriaBuilder cb);Class<T> getType();
}
abstract public class AbstractSpecification<T> implements Specification<T> {@Overridepublic boolean isSatisfiedBy(T t) {throw new NotImplementedException();}  @Overridepublic Predicate toPredicate(Root<T> poll, CriteriaBuilder cb) {throw new NotImplementedException();}@Overridepublic Class<T> getType() {ParameterizedType type = (ParameterizedType) this.getClass().getGenericSuperclass();return (Class<T>) type.getActualTypeArguments()[0];}
}

请暂时忽略带有神秘的getType()方法的AbstractSpecification <T>类(我们稍后再介绍)。

规范的中心部分是isSatisfiedBy()方法,该方法用于检查对象是否满足规范。 toPredicate()是在本示例中使用的另一种方法,用于以javax.persistence.criteria.Predicate实例的形式返回约束,该约束可用于查询数据库。

对于每个约束,我们创建一个新的规范类,该类扩展AbstractSpecification <T>并实现isSatisfiedBy()和toPredicate()。

检查轮询是否正在运行的规范实现如下所示:

public class IsCurrentlyRunning extends AbstractSpecification<Poll> {@Overridepublic boolean isSatisfiedBy(Poll poll) {return poll.getStartDate().isBeforeNow() && poll.getEndDate().isAfterNow() && poll.getLockDate() == null;}@Overridepublic Predicate toPredicate(Root<Poll> poll, CriteriaBuilder cb) {DateTime now = new DateTime();return cb.and(cb.lessThan(poll.get(Poll_.startDate), now),cb.greaterThan(poll.get(Poll_.endDate), now),cb.isNull(poll.get(Poll_.lockDate)));}
}

在isSatisfiedBy()中,我们检查传递的对象是否与约束匹配。 在toPredicate()中,我们使用JPA的CriteriaBuilder构造谓词。 稍后,我们将使用结果谓词实例来构建用于查询数据库的CriteriaQuery。

用于检查民意调查是否受欢迎的规范看起来类似:

public class IsPopular extends AbstractSpecification<Poll> {@Overridepublic boolean isSatisfiedBy(Poll poll) {return poll.getLockDate() == null && poll.getVotes().size() > 100;}  @Overridepublic Predicate toPredicate(Root<Poll> poll, CriteriaBuilder cb) {return cb.and(cb.isNull(poll.get(Poll_.lockDate)),cb.greaterThan(cb.size(poll.get(Poll_.votes)), 5));}
}

如果现在我们要测试Poll实例是否符合以下约束之一,则可以使用我们新创建的规范:

boolean isPopular = new IsPopular().isSatisfiedBy(poll);
boolean isCurrentlyRunning = new IsCurrentlyRunning().isSatisfiedBy(poll);

为了查询数据库,我们需要扩展DAO /存储库以支持规范。 看起来可能如下所示:

public class PollRepository {private EntityManager entityManager = ...public <T> List<T> findAllBySpecification(Specification<T> specification) {CriteriaBuilder criteriaBuilder = entityManager.getCriteriaBuilder();// use specification.getType() to create a Root<T> instanceCriteriaQuery<T> criteriaQuery = criteriaBuilder.createQuery(specification.getType());Root<T> root = criteriaQuery.from(specification.getType());// get predicate from specificationPredicate predicate = specification.toPredicate(root, criteriaBuilder);// set predicate and execute querycriteriaQuery.where(predicate);return entityManager.createQuery(criteriaQuery).getResultList();}
}

在这里,我们最终使用在AbstractSpecification <T>中实现的getType()方法来创建CriteriaQuery <T>和Root <T>实例。 getType()返回由子类定义的AbstractSpecification <T>实例的泛型类型。 对于IsPopular和IsCurrentlyRunning,它将返回Poll类。 如果没有getType(),我们将不得不在我们创建的每个规范的toPredicate()内创建CriteriaQuery <T>和Root <T>实例。 因此,减少规范内的样板代码只是一个小帮手。 如果您想出更好的方法,请随时用自己的实现替换它。

现在,我们可以使用我们的存储库来查询数据库以查找与特定规范匹配的民意测验:

List<Poll> popularPolls = pollRepository.findAllBySpecification(new IsPopular());
List<Poll> currentlyRunningPolls = pollRepository.findAllBySpecification(new IsCurrentlyRunning());

在这一点上,规范是唯一包含约束定义的组件。 我们可以使用它来查询数据库或检查对象是否满足必需的规则。

但是,仍然存在一个问题:我们如何结合两个或更多个约束? 例如,我们想在数据库中查询所有仍在运行的流行民意调查。

答案是复合设计模式的一种变化,称为复合规格。 使用复合规范,我们可以以不同方式组合规范。

要查询数据库中所有正在运行和流行的池,我们需要使用逻辑和操作将isCurrentlyRunningisPopular规范结合在一起。 让我们为此创建另一个规范。 我们将其命名为AndSpecification:

public class AndSpecification<T> extends AbstractSpecification<T> {private Specification<T> first;private Specification<T> second;public AndSpecification(Specification<T> first, Specification<T> second) {this.first = first;this.second = second;}@Overridepublic boolean isSatisfiedBy(T t) {return first.isSatisfiedBy(t) && second.isSatisfiedBy(t);}@Overridepublic Predicate toPredicate(Root<T> root, CriteriaBuilder cb) {return cb.and(first.toPredicate(root, cb), second.toPredicate(root, cb));}@Overridepublic Class<T> getType() {return first.getType();}
}

AndSpecification是从其他两个规范中创建的。 在isSatisfiedBy()和toPredicate()中,我们通过逻辑和运算返回两个规范的结果。

我们可以像这样使用我们的新规范:

Specification<Poll> popularAndRunning = new AndSpecification<>(new IsPopular(), new IsCurrentlyRunning());
List<Poll> polls = myRepository.findAllBySpecification(popularAndRunning);

为了提高可读性,我们可以在规范接口中添加and()方法:

public interface Specification<T> {Specification<T> and(Specification<T> other);// other methods
}

并在我们的抽象实现中实现它:

abstract public class AbstractSpecification<T> implements Specification<T> {@Overridepublic Specification<T> and(Specification<T> other) {return new AndSpecification<>(this, other);}// other methods
}

现在,我们可以使用and()方法链接多个规范:

Specification<Poll> popularAndRunning = new IsPopular().and(new IsCurrentlyRunning());
boolean isPopularAndRunning = popularAndRunning.isSatisfiedBy(poll);
List<Poll> polls = myRepository.findAllBySpecification(popularAndRunning);

在需要时,我们可以轻松地通过其他组合规范(例如OrSpecification或NotSpecification)进一步扩展该规范。

结论

使用规范模式时,我们将业务规则移到单独的规范类中。 通过使用复合规范,可以轻松组合这些规范类。 通常,规范可以提高可重用性和可维护性。 另外,规格可以轻松进行单元测试。 有关规范模式的更多详细信息,我推荐Eric Evans和Martin Fowler 撰写的这篇文章 。

  • 您可以在GitHub上找到此示例项目的源代码。

参考: Java: mscharhag,Programming and Stuff博客上的JCG合作伙伴 Michael Scharhag的JPA使用规范模式 。

翻译自: https://www.javacodegeeks.com/2014/01/java-using-the-specification-pattern-with-jpa.html

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

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

相关文章

python财经数据接口包Tushare pro的入门及简单使用方式(大数据,股票数据接口)...

最近在做一个项目&#xff0c;需要用到股票的数据&#xff0c;我在网上查了很久&#xff0c;最终发现在股票数据上面还是tushare比较专业&#xff0c;而且对于将来做金融行业的大数据这一块的&#xff0c;tushare绝对是你的一个好帮手&#xff0c;所以下面我就简单介绍一下。 一…

java ean13 条形码_【教程】Spire.Barcode 教程:如何在C#中创建EAN-13条码

基于UPC-A标准的EAN-13在世界范围内用于标记零售商品。 13位EAN-13号码由四部分组成&#xff1a;国家代码 - 2或3位数字制造商代码 - 5至7位数字产品代码 - 3至5位数字检查数字 - 最后一位数字代码演示&#xff1a;Step 1: 创建一个BarcodeSettings实例。BarcodeSettings setti…

C# 中串口通信 serialport1.DataReceived 函数无法触发或者出发延时等等问题解决方法...

C# 中串口通信 serialport1.DataReceived 函数无法触发或者出发延时等等问题解决方法 原文:C# 中串口通信 serialport1.DataReceived 函数无法触发或者出发延时等等问题解决方法曾经这个问题困扰我多天最后查资料一大堆&#xff0c;最后终于解决了&#xff0c;看到很多人做C#串…

Module的加载实现

烂笔头开始记录小知识点啦&#xff5e; 浏览器要加载 ES6模块&#xff0c;&#xff1a; <script type"module" src"./foo.js"></script> 异步加载&#xff0c;相当与defer属性。可以另外设置async属性。ES6 模块也允许内嵌在网页中&#xff0…

python 线型_CCF 202006-1 线性分类器 python

题目&#xff1a;线性分类器(line)【题目描述】考虑一个简单的二分类问题——将二维平面上的点分为A和B两类。训练数据包含n个点&#xff0c;其中第i个点(1≤i≤n)可以表示为一个三元组(x,y,type)&#xff0c;即该点的横坐标、纵坐标和类别。在二维平面上&#xff0c;任意一条直…

[Swift]LeetCode682. 棒球比赛 | Baseball Game

★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★➤微信公众号&#xff1a;山青咏芝&#xff08;shanqingyongzhi&#xff09;➤博客园地址&#xff1a;山青咏芝&#xff08;https://www.cnblogs.com/strengthen/&#xff09;➤GitHub地址&a…

Java调试器–权威的工具列表

Java调试是一个复杂的空间。 调试器的类型很多&#xff0c;并且有很多工具可供选择。 在此页面中&#xff0c;我们将介绍7种类型的调试器之间的区别&#xff0c;并查看每个类别中的主要工具&#xff0c;以帮助您为正确的工作选择正确的工具。 以下是我们涵盖的调试器类型&…

java项目中多个定时器_在java项目中如何使用Timer定时器

在java项目中如何使用Timer定时器发布时间&#xff1a;2020-11-16 16:36:16来源&#xff1a;亿速云阅读&#xff1a;97作者&#xff1a;Leah在java项目中如何使用Timer定时器&#xff1f;很多新手对此不是很清楚&#xff0c;为了帮助大家解决这个难题&#xff0c;下面小编将为大…

慎使用sql的enum字段类型

在sql的优化中&#xff0c;会有同学提到一点&#xff1a;使用enum字段类型&#xff0c;代替其他tinyint等类型。以前这也是不少人喜欢优化的&#xff0c;但是现在细想&#xff0c;是非常不合理的。 优点&#xff1a; 1.可以设置区间范围&#xff0c;比如设置性别&#xff1a;1男…

js对HTML字符转义与反转义

注意&#xff1a; 在编写html时&#xff0c;经常需要转义&#xff0c;才能正常显示在页面上。 并且&#xff0c;还可以防止xss。 解决方案&#xff1a; 一&#xff0c; 使用正则&#xff1a; 使用正则转码&#xff1a; var value document.getElementById(input).value.t…

python三维数据图_matplotlib中三维数据的热图

我想用我的三维数据生成一张热图。在我已经能够用这些数据绘制出trisurf。在有人能帮我制作热图吗&#xff1f;我看到了在线教程&#xff0c;但是它们对3D来说都很复杂&#xff0c;我在这个网站上找到了一个在matplotlib中生成带有散点的热图&#xff0c;但是它只有2D数据。在我…

区分基于Ant目标的Gradle任务

在我的博客文章《 从Ant Build演变Gradle构建&#xff1a;导入Ant构建文件》中 &#xff0c;我演示了如何使用Gradle内置的基于AntBuilder的Ant支持在Gradle构建中导入Ant目标。 然后&#xff0c;可以将这些Ant目标作为Gradle任务进行访问&#xff0c;并与Gradle构建直接引入的…

java显示长度和第一个字符_从Java字符串中以长度1的字符串返回的第一个字母的最佳方法是什么?...

假设以下内容&#xff1a;String example "something";String firstLetter "";以下分配方式firstLetter可能会影响性能&#xff0c;请注意是否存在差异&#xff1b; 哪个最好&#xff0c;为什么&#xff1f;firstLetter String.valueOf(example.charAt(…

sequelize difference between hasone and hasmany

Query is equal, hasone return the first instance from the collection,hasmany return the whole collection. 转载于:https://www.cnblogs.com/ybleeho/p/9772902.html

localStorage封装借口store.js的使用

localstorage是 HTML5 提供的在客户端存储数据的新方法&#xff0c;主要作用是将数据保存在客户端中&#xff0c;并且数据是永久保存的&#xff0c;除非人为干预删除。localstorage 的局限 1、只有版本较高的浏览器中才支持 localstorage2、localStorage的值的类型限定为strin…

AmazonSQS和Spring用于消息传递队列

下一篇文章将演示如何将Spring JMS模板和DLMC与AmazonSQS API一起使用&#xff0c;以放置消息队列。 我为什么要使用Amazon SQS&#xff1f; 易于配置 跨平台支持 从您的自我冗余&#xff0c;连带和扩展方面的烦恼中赚钱。 为什么我不使用Amazon SQS&#xff1f; 如果延迟…

3.3-3.9 周记

3.3-3.10 1. NIM游戏 百度链接&#xff1a;https://baike.baidu.com/item/Nim%E6%B8%B8%E6%88%8F/6737105?fraladdin 定义&#xff1a; P局面&#xff1a;先手必败N局面&#xff1a;先手必胜P局面的所有子局面都是N局面。N局面的子局面中必有一个是P局面 性质&#xff1a;\(a_…

关于类的使用的几个关键

类的定义和声明必须放在main函数前 如果类中只有申明类而没有定义&#xff0c;则只能定义指针&#xff1a;Test *test&#xff1b;如果不定义类而仅仅声明类的话&#xff0c;当使用Test test时&#xff0c;编译器只知道Test是个class&#xff0c;但留多大空间&#xff1f;怎么初…

java判断对象已死_Java的JVM判断对象已死的基本算法分析

jvm中有各种的垃圾收集器&#xff0c;每个收集器都有各自的算法。但是一切的根本都需要找到找到应该被消除的对象&#xff0c;理解如何找到死亡对象才是理解垃圾收集器的基础。01两个基本算法a、引用记数法&#xff1a;对象中加一个引用计数器&#xff0c;每次被引用计数器加一…

Java开发中更多常见的危险信号

在《 Java开发中的常见危险信号》一文中&#xff0c;我研究了一些不一定本身就是错误或不正确的做法&#xff0c;但它们可能表明存在更大的问题。 这些“红色标记”类似于“代码气味”的概念&#xff0c;我在这篇文章中引用的某些特定“红色标记”被称为“代码气味”。 正如我在…