Drools开源业务规则引擎(二)- Drools规则语言(DRL)


文章目录

    • 1.DRL文件的组成:
    • 2.package
    • 3.import
    • 4.function
    • 5.query
    • 6.declare
    • 7.global
    • 8.rule
      • 8.1.规则属性
      • 8.2.LHS
        • 8.2.1.语法格式
        • 8.2.2.运算符优先级
        • 8.2.3.特殊的运算符
          • 1.`matches`, `not matches`
          • 2.`contains`, `not contains`
          • 3.`memberOf`, `not memberOf`
          • 4.`in`, `notin`
          • 5.`soundslike`
          • 6.`str`
          • 7.`from`
      • 8.3.RHS
        • 8.3.1.`set`
        • 8.3.2.`modify`
        • 8.3.3.`update`
        • 8.3.4.`insert`
        • 8.3.5.`delete`
        • 8.3.6.`drools.halt()`


1.DRL文件的组成:

在官方文档中(链接地址 https://docs.drools.org/7.74.1.Final/drools-docs/html_single/index.html#_droolslanguagereferencechapter),DRL的组成如下:

package

import

function  // Optional

query  // Optional

declare   // Optional

global   // Optional

rule "rule name"
    // Attributes
    when
        // Conditions
    then
        // Actions
end

rule "rule2 name"

...

以上内容,只是梳理了DRL文件中组成的元素及其顺序(顺序非强制性要求,这是这样便于内容阅读)。

2.package

类似于 Java 中的 package,声明规则的唯一命名空间(包)。包必须有一个名称空间,并使用包名称的标准 Java 约定进行声明;即没有空格,这与允许空格的规则名称不同。根据元素的顺序,它们可以以任何顺序出现在规则文件中,但package语句除外,它必须位于文件的顶部。在任何情况下,分号都是可选的。

任何规则属性(见“规则属性”一节)也可以在包级别编写,取代该属性的默认值。修改后的默认值仍可能被规则中的属性设置所替换。

示例

// 带分号
package com.test;
// 不带分号
package com.test

3.import

类似于 Java 中的 import。DRL文件中在规则中需要导入使用的对象的完全限定路径和类型名称。Drools引擎自动从与DRL包同名的Java包和Java.lang包中导入类。

示例

// 不需要导入,已自动导入
// import java.lang.String;

import com.a.b.DroolsObject;

4.function

类似于 Java 中的 method。可以将重复代码或者逻辑性代码封装成一个函数,或者将工具/辅助类中的静态方法导入为函数,然后在规则LHS/RHS部分中按名称使用该函数。

示例

// 导入静态方法/函数
import static org.example.applicant.MyFunctions.hello;

// 函数1
function boolean ageJudge(Integer i){
    if (i > 10){
      return true;
    } else {
      return false;
    }
}

// 使用函数
rule "age"
enabled true
when
    eval(ageJudge(10));
then
end

5.query

查询在工作内存中相关的事实对象。查询的名称对于 KieBase 是全局的,因此对于该项目中的所有其他规则的查询必须是唯一的。ksession.getQueryResults("name") 获取查询结果,其中“name”是查询名称。

示例

package drools.drl

// 导入输入类型
import com.ahao.project.input.UserIn;
global com.ahao.project.output.UserOut output;


query "age>10"
    $in:UserIn(age > 10)
end

获取结果

QueryResults queryResults = kieSession.getQueryResults("age>10");
Iterator<QueryResultsRow> iterator = queryResults.iterator();
while (iterator.hasNext()) {
    QueryResultsRow next = iterator.next();
    log.info("查询结果:{}",next.get("$in"));
}

// [main] INFO DroolsTest2 - 查询结果:UserIn(age=35)
// [main] INFO DroolsTest2 - 查询结果:UserIn(age=45)

6.declare

声明类型。用于在DRL声明一个新的类型。

示例

package drools.drl

import com.ahao.project.input.UserIn

// 声明一个Person类型
declare Person
    name:String
    age:Integer
end

rule "age"
enabled true
when
    $in:UserIn(age > 10)
then
    Person person = new Person();
    person.setAge($in.getAge());
    person.setName("命中规则");
    System.out.println(person);
end

结果
在这里插入图片描述

7.global

全局变量。DRL文件中的全局变量通常为规则提供数据或服务,如在then中使用的应用程序服务(service),并从规则返回数据。

示例

// 全局变量的定义
global com.ahao.project.output.UserOut output;
// 可以有多个
// global com.ahao.project.output.UserOut output222;

注意:

执行规则时,需要先设置全局变量对象

kieSession.setGlobal("output", output);
// 然后再 kieSession.fireAllRules()

8.rule

rule语法结构如下图

rule

8.1.规则属性

属性默认值描述
salience0规则优先级。在同一个 activation-group 中,具有较高优先级(值越大)的规则优先执行
示例: salience 10
enabledtrue是否启用该规则
示例: enabled true
date-effectivenull只有当前日期时间在 date-effective 属性之时,才能激活该规则
示例: date-effective "4-Sep-2018"
date-expiresnull只有当前日期时间在 date-expires 属性之时,才能激活该规则
示例: date-expires "4-Oct-2018"
no-loopfalse是否可循环。某些情况下(update函数等),规则可以被再次激活
示例: no-loop true
agenda-groupdefault议程组。对议程进行分区,以提供对规则组的更多执行控制。只有获得焦点的议程组中的规则才能被激活。
示例: agenda-group "GroupName"
activation-groupnull激活组。一个激活组只能激活一个规则,即使其他规则体的LHS部分仍然为true也不会再被执行。
示例: activation-group "GroupName"
durationnull以毫秒为单位的持续时间。如果规则条件当前满足,则在该持续时间之后可以激活规则。
示例: duration 10000
timernull调度器。“时间间隔”或“cron”计时器定义
示例: timer ( cron:* 0/15 * * * ? ) (every 15 minutes)
calendarnull用于调度规则的 Quartz calendar
示例: calendars "* * 0-7,18-23 ? * *" (exclude non-business hours)
auto-focusfalse自动获取焦点,仅适用于议程组内的规则。选择该选项后,下次激活规则时,会自动将焦点指定给规则所分配的议程组:
示例:auto-focus true
lock-on-activefalse仅适用于规则流组或议程组内的规则。选择该选项后,下次规则的规则流组变为活动状态或规则的议程组收到焦点时,在规则流组不再活动或议程组失去焦点之前,无法再次激活规则。是no-loop属性的更强版本。
示例: lock-on-active true
ruleflow-groupnull规则流组。只有被关联的规则流激活这个组时,规则才能激活
示例: ruleflow-group "GroupName"
dialectmvel用来定义规则中要使用的语言类型
示例: dialect "java"

8.2.LHS

规则的when部分(也称为Left Hand Side(LHS))包含执行操作必须满足的条件。

8.2.1.语法格式

格式

// 格式如下(对象名称前的 $ 非必需的)
when
  $对象名称:事实对象的全限定类名(字段约束)
    
// 如果已经import了事实对象的全限定类名
when
  $对象名称:事实对象的类名(字段约束)  

示例

// 示例1
when
  $p:com.test.Person(age > 10 || name == "jack")
    
// 示例2
import com.test.Person;
when
  p:Person(age > 10 || name == "jack")
8.2.2.运算符优先级
操作类型符号Notes
嵌套或(null-safe)属性访问., .(), !.不是标准的Java语义
List 或者 Map 访问[]不是标准的Java语义
约束绑定(类型绑定):不是标准的Java语义
乘积*, /,%
加减+, -
位运算>>, >>>, <<
关系型<, <=, >, >=, instanceof
等于== !=使用equals()!equals()语义,非标准Java samenot same语义
非短路 AND&
非短路异运算^
非短路包含 OR``
逻辑运算 AND&&
逻辑运算 OR`
三元运算符? :
逗号分隔 AND,不是标准的Java语义

运算符==使用null-safe equals()语义,而不是通常的equals语义。例如,Person(firstName=="John") 类似于 java.util。Objects.equals(person.getFirstName(),“John”),并且由于"John"不为null,所以该模式也类似于"John".equals(person.get-FirstName)

以上运算符都基本上类似java中的运算符。接下会介绍一些特殊的运算符。

8.2.3.特殊的运算符
1.matches, not matches

判断与指定的Java正则表达式匹配或不匹配。

示例

Person( country matches "(USA)?\\S*UK" )
2.contains, not contains

数组或集合字段是否包含指定值。可以使用这些运算符来代替String.contains()和!String.contents()。

示例

// 姓名包含王
Person( name contains "王" )
// 假设一个人有多个手机号,phones:List<String>
// 手机号不包含12345678901
Person( phones not contains "12345678901" )
3.memberOf, not memberOf

是否为数组或集合的成员。

示例

// 手机号不包含12345678901
Person( "12345678901" not memberOf phones)
4.in, notin

指定约束中要匹配的多个可能值(复合值限制)。

示例

// 名称为 jack 或者 mary
Person( name in ("jack","mary"))
5.soundslike

验证单词是否具有与给定值几乎相同的发音(英语发音)。

示例

// firstName "Jon" or "John":
Person( firstName soundslike "John" )
6.str

验证字符串字段是以指定值开始,以指定值结束,或者字符串的长度。

示例

// 消息的以"R1"开始
Message( msg str[startsWith] "R1" )

// 消息的以"R2"结尾
Message( msg str[endsWith] "R2" )

// 消息的长度为17
Message( msg str[length] 17 )
7.from

获取数据源。

示例

// 从模式绑定(变量)中获取数据源
rule "rule1"
when
  Person( $personAddress : address )
  Address( zipcode == "11110" ) from $personAddress
then
end

// 从图形符号中获取数据源
rule "rule1"
when
  $p : Person()
  $a : Address( zipcode == "23920W" ) from $p.address
then
end
    
// 从迭代器中获取数据源
rule "rule1"
when
  $order : Order()
  $item  : OrderItem( value > 100 ) from $order.items
then
end

8.3.RHS

规则的then部分(也称为 Right Hand Side(RHS))。当满足规则的when部分时要执行的操作。操作的主要目的是在Drools引擎的工作内存中插入、删除或修改数据。有效的规则操作是小的、声明性的和可读的。如果需要在规则操作中使用命令式或条件式代码,请将规则划分为多个更小、更具声明性的规则。

8.3.1.set

赋值。

示例

$person.setName("jack");
8.3.2.modify

针对某个事实进行修改,并将更改通知Drools引擎。

语法

modify ( <fact-expression> ) {
    <expression>,
    <expression>,
    ...
}

示例

modify( Person ) {
        setAge( 100 ),
        setName ( "nick" )
}
8.3.3.update

指定要更新的字段和整个相关事实,并将更改通知Drools引擎。

语法

update ( <object, <handle> )
update ( <object> ) 

示例

person.setAge( 100 );
update( person );
8.3.4.insert

将一个新的事实插入Drools引擎的工作内存中。

语法

insert( new <object> );

示例

insert( new Person() );
8.3.5.delete

从Drools引擎中删除对象。

语法

delete( <object> );

示例

delete( person );
8.3.6.drools.halt()

终止规则执行。

示例

rule "end_rule"
enabled true
when
then
    drools.halt();
end

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

7.8作业

一、思维导图 二、 1】按值修改 2】按值查找&#xff0c;返回当前节点的地址 &#xff08;先不考虑重复&#xff0c;如果有重复&#xff0c;返回第一个&#xff09; 3】反转 4】销毁链表 //按值修改 int value_change(linklistptr H,datatype e,int value) {if(HNULL||empty(H…

推出PSoC™ 车规级4100S Max系列,(CY8C4147LDE 和 CY8C4147LDS)支持性能更强大的第五代CAPSENSE™技术

推出全新车规级PSoC™ 4100S Max系列。这一微控制器器件系列具有更佳的闪存密度、通用输入输出接口&#xff08;GPIO&#xff09;、CAN-FD和硬件安全性&#xff0c;扩展了采用CAPSENSE™技术的汽车车身/暖通空调&#xff08;HVAC&#xff09;和方向盘应用人机界面&#xff08;H…

Unity之Text组件换行\n没有实现+动态中英互换

前因:文本中的换行 \n没有换行而是打印出来了,解决方式 因为unity会默认把\n替换成\\n 面板中使用富文本这个选项啊 没有用 m_text.text = m_text.text.Replace("\\n", "\n"); ###动态中英文互译 using System.Collections; using System.Collections…

什么叫低频晶振?低频晶振最低频率能达到多少?低频晶振封装尺寸有哪些?

低频晶振指的是那些工作在较低频率范围内的晶体振荡器&#xff0c;通常这类振荡器的标称频率低于8MHz。这些晶振在各种电子设备中都有应用&#xff0c;尤其是在那些需要精确但不需要高频振荡的应用场景中&#xff0c;比如实时时钟(RTC)、低速串行通信接口(如UART、IC等)、以及一…

怎么才能选到好的猫咪主食冻干?公认顶尖优秀主食冻干总结

如今&#xff0c;主食冻干市场纷繁多样&#xff0c;质量水平却大相径庭。部分品牌盲目追求高营养值和利润增长&#xff0c;却忽略了猫咪健康饮食的本质需求&#xff0c;导致市场上充斥着以次充好、虚假标注日期等不法行为。更有甚者&#xff0c;部分产品未经权威第三方检测便匆…

提高LabVIEW软件的健壮性

提高LabVIEW软件的健壮性&#xff0c;即增强其在各种操作条件下的可靠性和稳定性&#xff0c;是开发过程中非常重要的一环。健壮的软件能够在面对意外输入、极端环境和系统故障时依然表现出色&#xff0c;确保系统的连续性和可靠性。以下是详细的方法和策略&#xff0c;从多个角…

【活动行】参与上海两场线下活动,教育生态行业赛总决赛活动和WAIC人工智能大会活动 - 上海活动总结

目录 背景决赛最后一公里领域范围 决赛作品AI智教相机辅导老师Copilot辅导老师Copilot雅思写作竞技场 优秀作品总结 背景 决赛 百度发起的千帆杯教育生态行业赛于2024年7月4日进行线下决赛&#xff0c;博主虽然没能进入决赛&#xff0c;但也非常荣幸能够以嘉宾身份到现场给进…

Leaflet【七】加载海量点数据

本文深入探讨了Leaflet在渲染海量点数据时面临的性能挑战&#xff0c;提出了一种创新的解决方案——利用leaflet-marker-canvas插件。传统的循环绘制Marker方式在数据量巨大时会导致明显的性能下降&#xff0c;而通过将点数据加入Canvas进行批量渲染&#xff0c;显著提高了绘图…

vite+vue3拍照上传到nodejs服务器

一:效果展示: 拍照效果 二:Nodejs后端接口代码: 三:前端完整代码:

【productj服务导出文件503问题】

设备服务只要导出文件&#xff0c;就报503&#xff0c;查看K8S发现服务重启 1. 复现问题1.1 问题复现频繁fullGC 宿主pod服务日志监控1.2 小内存复现 接口没啥问题&#xff0c;导出就会导致服务重启 1. 复现问题 当前uat环境配置&#xff1a; pod内存&#xff1a;2G JVM参数&a…

C++入门(C语言过渡)

文章目录 前言一、C关键字二、命名空间三、C输入&输出四、缺省参数五、函数重载六、引用七、inline八、nullptr总结 前言 C是一种通用的、高级的、静态类型的编程语言&#xff0c;它在20世纪80年代由丹尼斯里奇创建的C语言基础上发展而来。以下是C发展的一些重要里程碑。 1…

【目录】阅读须知!全博文、专栏大纲

首先要和大家说一下&#xff0c;博主的文章并不是想到哪里写到哪里&#xff0c;而是以整个大后端为主题&#xff0c;成体系的在写专栏&#xff0c;从和后端紧相关的计算机核心课程开始、到JAVA SE、JAVA EE、到数据库、MQ等各类中间件、再到业务场景、性能优化。当然也会涉及一…

Docker拉取失败,利用github将镜像推送到阿里云

背景 由于近期国内docker镜像地址失效&#xff08;2024年6月份开始&#xff09;&#xff0c;导致pull docker 镜像总是超时。 涉及到的网址和工具 https://github.com/tech-shrimp/docker_image_pusherhttps://hub.docker.com/search阿里云 GITHUB配置 fork https://githu…

FP5207+音频功率放大器的组合解决方案-适用于便携式音频播放器、无线耳机、智能音箱和车载音响系统等高质量音频输出需求的产品,以提高电池供电的效率和输出功率

随着消费者对智能家居的需求增长&#xff0c;智能音响市场成为重要增长点。同时&#xff0c;音响技术也在不断发展&#xff0c;音响及扬声器的功能和性能不断提升。 蓝牙音箱&#xff0c;这类音箱供电是以锂电池为主&#xff0c;一般选用内置升压的音频功放芯片&#xff0c;音响…

Vue框架引入

vue简介 1.1.vue是什么?Vue官网 英文官网: https://vuejs.org/中文官网: https://cn.vuejs.org/ vue是一套构建用户界面的渐进式javascript框架 构建用户界面:将我们手里拿到的数据通过某种办法变成用户可以看见的界面前端工程师的职责:就是在合适的时候发出合适的请求,然后…

Docker-11☆ Docker Compose部署RuoYi-Cloud

一、环境准备 1.安装Docker 附:Docker-02-01☆ Docker在线下载安装与配置(linux) 2.安装Docker Compose 附:Docker-10☆ Docker Compose 二、源码下载 若依官网:RuoYi 若依官方网站 鼠标放到"源码地址"上,点击"RuoYi-Cloud 微服务版"。 跳转至G…

微信如何快速回复信息呢?

时业务繁忙的时候可能会出现一大堆消息需要去回复&#xff0c;很多客户也会来问重复的问题&#xff0c;有时候回复消息也需要一个及时性&#xff0c;如果回复慢了有可能客户就跑了&#xff0c;那这个时候就会体现出自动回复的优势。 只要设置好一个关键词&#xff0c;只要对方…

基于React 实现井字棋

一、简介 这篇文章会基于React 实现井字棋小游戏功能。 二、效果演示 三、技术实现 import {useEffect, useState} from "react";export default (props) > {return <Board/> }const Board () > {let initialState [[, , ], [, , ], [, , ]];const [s…

【CW32F030CxTx StartKit开发板】利用超声波传感器实现智能灯控

目录 1、超声波传感器 2、硬件连线 3. 程序开发 3.1 超声波测距 3.2 LED控制 4. 演示视频 本文首发于21ic。 感谢21ic和武汉芯源提供的测试机会。 在上一篇帖子中介绍了CW32F030CxTxStartKit 评估板的环境构建。本次介绍如何利用超声波传感器实现人来灯亮&#xff0c;人…

Milvus lite start 及存储策略

背景 今天开始写下Milvus&#xff0c;为了方便&#xff0c;我直接使用的是 milvus-lite 版本&#xff0c;default 情况下&#xff0c;你可能不知道他到底将 db 存储到什么位置了。启动 default-server&#xff0c;看下Milvus 的start及存储逻辑 主逻辑 def start(self):sel…