DMN中的函数式编程:感觉就像再次重读我的大学课程一样

在本文中,我想分享有关DMN中的递归支持的有趣见解,并重点介绍FEEL语言的特定属性如何使功能编程结构能够在DMN中建模。

我们将从一个基本的示例开始,以演示FEEL语言和DMN构造的“商业友好”性质如何使我们能够解决一个通常不愉快的问题:递归函数的定义。 然后,我们将在FP土地中冒险,并且在FEEL / DMN的摇篮中,我们将欣赏功能构造最好的生物之一:Y Combinator。 最后,我们将再次被问到一个著名的问题:

使用纯工程方法,让我们立即深入研究问题!

基本递归示例

Drools DMN开源引擎允许在DMN商业知识模型节点中提供递归支持。 这使递归函数的建模非常容易, 这是在DMN中为递归函数建模时的推荐方法 :允许函数以其名称进行调用。

让我们看一个简单的示例:在DMN中对阶乘函数建模。

我们可以使用Kogito DMN编辑器,并如下定义DRD:

使用“事实”业务知识模型(简称BKM)节点以递归方式定义实际的阶乘函数为:

我们可以注意到,该函数像其他任何普通函数一样调用自身
递归函数,唯一的区别是它被定义为DMN Boxed Expression的一部分; 该函数的名称由BKM节点使用框式表达式构造“ fac”定义,然后该函数的主体进行引用并将其自身作为FEEL表达式“ fac(n-1)”的一部分进行调用。

我们可以使用此BKM来计算输入数据节点传递的实际结果,作为“计算阶乘”决策的一部分,如下所示:

这可以很好地工作并给出预期的结果:

{ 我的电话:3 fac:函数fac(n) 计算阶乘:6 }

关于柯里

DMN以及更重要的是FEEL语言允许定义和调用咖喱函数。

这使我们可以在FEEL中编写如下内容:

{f:function(a)function(b)a + b,r:f(1)(2)}

哪里:

  • 我们定义了一个touch:context有2个条目
  • 第一个条目名为“ f”并定义了一个咖喱函数:一个参数“ a”的函数,一旦被调用,将返回一个参数“ b”的函数,一旦被调用,将返回a + b的和
  • 后一个名为“ r”的条目以a = 1和b = 2调用咖喱函数。

尽管这可能是一个看起来很奇怪的FEEL表达式,但是一旦执行r = 3,我们就不会感到惊讶。

我们可以使用DMN Boxed Expression构造等效地做:

这是一个名为“咖喱和”的BKM节点; DMN可调用一个参数“ a”,一旦被调用,将返回一个参数“ b”的函数,该函数一旦被调用,将返回a + b之和。

同样,一旦执行我们就不会感到惊讶 咖喱求和(1)(2)= 3

Y组合器:无递归支持的递归

让我们回头看一下前面的递归函数示例。 我们忽略了以下事实:在DMN中,函数实际上是否可以通过其名称进行自身调用:DMN规范未明确支持此功能,但也未明确禁止它。 换句话说,没有正式指定递归支持。

如果我们仍然需要定义递归函数,但又发现道路仍在建设中,缺少正式的递归支持,该怎么办? 我们可以使用一种称为“ Y Combinator ”的功能设备,该设备允许匿名函数实现递归,而不必依靠自身(不存在)的名称进行自我调用。

让我们看一个例子; 我们可以在DMN中定义Y组合器,如下所示:

它可能是一个看起来很奇怪的函数:)让我们假设它是为我们定义的,我们可以使用它。

我们可以使用它来重新定义阶乘计算,如下所示:

我们可以注意到,“ fac”函数定义的主体在总体上是相同的; 但是,它不再是一个通过名称调用自身的函数:在函数主体中没有任何对“ fac(…)”的调用的痕迹!

自然,仍然会有某种形式的递归发生,但是这次是利用闭包范围内的参数名称:“ f”。 结果按预期工作: fac(3)= 6

我们可以看一下另一个示例,该示例使用DMN中的Y组合器定义斐波那契序列:

我们再次注意到,在函数体中没有对“ fib(…)”的调用,但是由于使用了Y组合器,因此可以执行斐波那契数列的递归计算。

再次,结果按预期工作: fib(5)= [1、2、3、5]

为了获得更多乐趣,我们可以使用DMN Boxed Expression形式重新定义Y组合器。 这是一个有趣的练习,了解如何在其盒装变量中应用闭包。 Y组合器的定义可以重构为:

这将再次产生相同的预期和正确结果。

对于(额外(额外的乐趣)),我们可以在单个FEEL表达式中再次重新定义Y组合器以计算例如4的阶乘:

{Y:function(f)(function(x)x(x))(function(y)f(function(x)y(y)(x))),fac:Y(function(f)function(n)如果n> 1,则n * f(n-1)否则1),fac4:fac(4)} .fac4

结果不出所料:24。

结论

在本文中,我们看到了DMN中递归的基本示例,并且如何在引擎中利用递归支持非常简单; 支持引擎递归支持是我们建议实现递归DMN的方法:给函数命名,并在函数主体中使用该名称来调用自身。 在该示例中,我们将函数命名为“ fac”,然后在函数本身的主体中调用了“ fac(…)”。

这种方法非常实用,易于在DMN中建模,并且效果很好。

我们还看到了DMN和FEEL如何确实支持咖喱函数定义和调用。 FEEL(也是)一种功能语言; 所有这些属性使我们能够在DMN中定义并使用Y Combinator,这是一种无需递归支持即可实现递归的功能性设备!

我个人发现这些练习对于在DMN中应用函数式编程概念非常有趣,同时确保引擎按预期运行。 我要特别感谢我的同事Edoardo Vacchi和Luca Molteni在讨论Y组合器和Currying功能时所给予的支持。

对DMN感兴趣?

如果您以前不了解DMN,那么您会发现这篇文章很有趣,但是想要对DMN标准进行温和介绍,我们提供了有关DMN的正确的崩溃课程,您可以通过以下网址免费获得: http://learn-dmn-in-15-minutes.com

翻译自: https://www.javacodegeeks.com/2020/04/functional-programming-in-dmn-it-feels-like-recursing-my-university-studies-again.html

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

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

相关文章

手游极品飞车无限狂飙链接服务器失败,极品飞车无极限无法联网是什么原因 联网失败原因分析及解决方法...

有些玩家对于极品飞车无极限游戏中无法联网的问题而困扰,应该怎么解决呢?下面42824小小编就把方法分享给大家!一、极品飞车无极限游戏无法联网原因及解决方法1、网络连接不稳定推荐在wifi的情况下进行游戏,如果是3G网的话很容易会…

mysql索引命名规范_mysql使用规范-索引规范

(1)单张表中索引数量不超过5个。(2)单个索引中的字段数不超过5个。(3)索引名必须全部使用小写。(4)非唯一索引按照“idx_字段名称[_字段名称]”进用行命名。例如idx_age_name。(5)唯一索引按照“uniq_字段名称[_字段名称]”进用行命名。例如uniq_age_name。(6)组合索引建议包含…

junit规则_jUnit:规则

junit规则规则在测试,测试用例或测试套件周围增加了特殊处理。 他们可以对该类中的所有测试执行通用的其他验证,并发运行多个测试实例,在每个测试或测试用例之前设置资源,然后在之后拆除它们。 该规则可以完全控制将要应用到的测…

mysql中创建唯一索引的关键字_mysql中唯一索引的关键字是什么

mysql中唯一索引的关键字是unique index。创建唯一索引可以避免数据出现重复。唯一索引可以有多个,但索引列的值必须唯一,索引列的值允许有空值。创建唯一索引可以使用关键字UNIQUE随表一同创建。mysql中唯一索引的关键字是unique index。(推荐教程&…

奇迹觉醒qq服务器比微信少,十年内最大的奇迹!功能比QQ还少的微信为什么能成功?...

今天,微信迎来了自己2021年的第一次「翻车」——2021年1月18日下午2点前后,「由于系统抖动原因」部分微信用户无法及时收取微信消息。截止下午3点19分,故障已被修复。其实微信曾面临过几次信息服务中断的事故:2013年,微…

从fork-join /线程池调用的Singelton bean中的访问spring请求范围缓存

问题: 启用了Spring且将范围设置为Request的缓存需要由不在请求范围内的singleton bean访问。 解: Spring使您能够创建缓存,该缓存为请求范围保留数据。 例如 import org.springframework.cache.concurrent.ConcurrentMapCache; import org…

linux终止mysql进程_Ubuntu Linux下定时监测MySQL进程终止时自动重启的方法

前言最近发现MySQL服务隔三差五就会挂掉,导致我的网站和爬虫都无法正常运作。自己的网站是基于MySQL,在做爬虫存取一些资料的时候也是基于MySQL,数据量一大了,MySQL它就有点受不了了,时不时会崩掉,虽然我自…

系统错误null是什么意思_为什么NULL是错误的?

系统错误null是什么意思Java中NULL用法的简单示例: public Employee getByName(String name) {int id database.find(name);if (id 0) {return null;}return new Employee(id); }这种方法有什么问题? 它可能返回NULL而不是对象-这是错误的。 在面向对…

mysql数据库版本不同_mysql数据库版本不同所引起的问题

1.sql_mode不同所引起的问题mysql5.7 ERROR 1055 (42000): Expression #1 of SELECT list is not in GROUP BY clause and contains nonaggregated column lhh.lhh.id which is not functionally dependent on columns in GROUP BY clause; this is incompatible with sql_mode…

线性搜索或顺序搜索算法在Java中如何工作? 示例教程

大家好,我之前谈到了二进制搜索算法的工作原理,并分享了在Java中实现二进制搜索的代码。 在那篇文章中,有人问我是否还存在其他搜索算法? 如果数组中的元素未排序,又如何使用二进制搜索算法,该如何搜索呢&a…

junit规则_JUnit规则

junit规则介绍 在本文中,我想展示一个示例,说明如何使用JUnit Rule简化测试。 最近,我继承了一个相当复杂的系统,并未对所有内容进行测试。 甚至经过测试的代码也很复杂。 通常,我看到缺乏测试隔离。 (我将…

mysql server 5.0安装教程_MySQL Server 5.0安装教程

运行MySQL Server 5.0安装程序“setup.exe”,出现如下界面:安装向导启动,按“Next”继续:选择安装类型,为了方便熟悉安装过程,我们选择“Custom”。按“Next”继续:在“Developer Components”上…

Kubernetes集群上的Apache Ignite和Spring第2部分:Kubernetes部署

以前,我们已经成功创建了第一个由Apache Ignite支持的Spring boot Application。 在此博客上,我们将重点介绍Kubernetes方面需要做的事情,以便能够启动我们的应用程序。 如先前博客所述,我们需要制定我们的Kubernetes RBAC策略。…

centos6.5 安装多个mysql_在centos6,5(64位)系统安装多实例mysql5.6

首先你安装个单实例的mysql试一试一 检查你电脑之前是否装了mysqlrpm -qa | grep mysql这条命令只是查看你系统是否使用过yum或者rpm包安装mysql,对用源码包安装的mysql是查不到的,所以基本用不到二 安装编译所需的包yum -y install make gcc-c cmake bi…

hibernate jpa_JPAHibernate替代方案。 如果JPA或Hibernate对于我的项目而言不够好,该怎么办?...

hibernate jpa你好!你好吗? 今天我们将讨论不建议使用JPA / Hibernate的情况。 在JPA领域之外,我们还有哪些选择? 我们将谈论的是: JPA /Hibernate问题 解决一些JPA /Hibernate问题的方法 选择此处描述的框架的标准…

mysql中常见查询表_MySQL中常见查询

1 --1、查询“001”课程比“002”课程成绩高的所有学生的学号;2 SELECT a.s FROM sc a,sc b WHERE a.sb.s AND a.c1 AND b.c2 AND a.score >b.score;3 --2、查询平均成绩大于60分的同学的学号和平均成绩;4 SELECT student.s,avg(score) FROM student,…

约束流–没有Drools规则语言的现代Java约束

传统上,要使用OptaPlanner进行扩展,您必须学习DRL。 不再。 借助受Java 8 Streams和SQL启发的新Constraints Streams API,您现在可以用Java (或Kotlin或Scala) 编写约束,并且仍然可以从增量计算中受益。 在…

mysql数据库杀掉堵塞_Mysql解决USE DB堵塞详解

遇到故障,我们往往想的是如何解决这个故障,而不是从故障的根本去思考出现这个故障的原因?这样的结果,只能使我们得到了鱼,失去了渔。今天,我们就来分享一个由USE DB堵塞故障引发的思考案例。故障描述今天一…

java拦截器项目应用_使用拦截器分析Java EE应用程序的性能下降/提高

java拦截器项目应用在开发具有某些性能要求的Java EE应用程序时,必须在每个发行版之前验证是否满足这些要求。 您可能会想到,哈德森的一项工作每天晚上在某些特定的硬件平台上执行一系列测试测量。 您可以检查已实现的时间并将它们与给定的要求进行比较…

iis web.config 配置 经典模式_django部署在iis下,webconfig错误

django部署在iis下,webconfig错误错误原因:iis7以后,web.config管理机制更安全了默认情况下,会锁住配置项,不许修改怎么办?如何求解以上问题呢?D:django_websiteshello>%windir%C:Windows 不…