当前位置: 首页 > news >正文

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的组成如下:

packageimportfunction  // Optionalquery  // Optionaldeclare   // Optionalglobal   // Optionalrule "rule name"// Attributeswhen// Conditionsthen// Actions
endrule "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
wheneval(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.drlimport com.ahao.project.input.UserIn// 声明一个Person类型
declare Personname:Stringage:Integer
endrule "age"
enabled true
when$in:UserIn(age > 10)
thenPerson 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;
whenp: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"
whenPerson( $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
thendrools.halt();
end

相关文章:

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

文章目录 1.DRL文件的组成&#xff1a;2.package3.import4.function5.query6.declare7.global8.rule8.1.规则属性8.2.LHS8.2.1.语法格式8.2.2.运算符优先级8.2.3.特殊的运算符1.matches, not matches2.contains, not contains3.memberOf, not memberOf4.in, notin5.soundslike6…...

PTA甲级1005:Spell It Right

错误代码&#xff1a; #include<iostream> #include<vector> #include<unordered_map> using namespace std;int main() {unordered_map<int, string> map {{0, "zero"}, {1, "one"}, {2, "two"}, {3, "three&qu…...

Vue笔记11-Composition API的优势

Options API存在的问题 使用传统Options API中&#xff0c;新增或者修改一个需求&#xff0c;就需要分别在data&#xff0c;methods&#xff0c;computed里修改&#xff0c;而这些选项分布在代码的各个地方&#xff0c;中间还穿插着其他Optional API&#xff0c;如果代码量上来…...

rancher管理多个集群

一、rancher部署 单独部署到一台机器上&#xff0c;及独立于k8s集群之外&#xff1a; 删除所有yum源&#xff0c;重新建yum源&#xff1a; # 建centos7.9的yum源 # cat CentOS-Base.repo # CentOS-Base.repo # # The mirror system uses the connecting IP address of the …...

某大会的影响力正在扩大,吞噬了整个数据库世界!

1.规模空前 你是否曾被那句“上有天堂&#xff0c;下有苏杭”所打动&#xff0c;对杭州的湖光山色心驰神往&#xff1f;7月&#xff0c;正是夏意正浓的时节&#xff0c;也是游览杭州的最佳时期。这座古典与现代交融的城市将迎来了第13届PostgreSQL中国技术大会。作为全球数据库…...

PostgreSQL主从复制:打造高可用数据库架构的秘籍

PostgreSQL主从复制&#xff1a;打造高可用数据库架构的秘籍 在当今的数字化时代&#xff0c;数据的安全性和可靠性是企业最为关注的问题之一。PostgreSQL作为一种强大的开源关系型数据库管理系统&#xff0c;提供了多种高可用性解决方案&#xff0c;其中主从复制是最为常用的…...

Fast R-CNN(论文阅读)

论文名&#xff1a;Fast R-CNN 论文作者&#xff1a;Ross Girshick 期刊/会议名&#xff1a;ICCV 2015 发表时间&#xff1a;2015-9 ​论文地址&#xff1a;https://arxiv.org/pdf/1504.08083 源码&#xff1a;https://github.com/rbgirshick/fast-rcnn 摘要 这篇论文提出了一…...

视觉语言模型:融合视觉与语言的未来

1. 概述 视觉语言模型&#xff08;Vision-Language Models, VLMs&#xff09;是能够同时处理和理解视觉&#xff08;图像&#xff09;和语言&#xff08;文本&#xff09;两种模态信息的人工智能模型。这种模型结合了计算机视觉和自然语言处理的技术&#xff0c;使得它们能够在…...

【CSAPP】-linklab实验

目录 实验目的与要求 实验原理与内容 实验步骤 实验设备与软件环境 实验过程与结果&#xff08;可贴图&#xff09; 实验总结 实验目的与要求 1.了解链接的基本概念和链接过程所要完成的任务。 2.理解ELF目标代码和目标代码文件的基本概念和基本构成 3.了解ELF可重定位目…...

UE C++ 多镜头设置缩放 平移

一.整体思路 首先需要在 想要控制的躯体Pawn上&#xff0c;生成不同相机对应的SpringArm组件。其次是在Controller上&#xff0c;拿到这个Pawn&#xff0c;并在其中设置输入响应&#xff0c;并定义响应事件。响应事件里有指向Pawn的指针&#xff0c;并把Pawn的缩放平移功能进行…...

代码随想录Day69(图论Part05)

并查集 // 1.初始化 int fa[MAXN]; void init(int n) {for (int i1;i<n;i)fa[i]i; }// 2.查询 找到的祖先直接返回&#xff0c;未进行路径压缩 int.find(int i){if(fa[i] i)return i;// 递归出口&#xff0c;当到达了祖先位置&#xff0c;就返回祖先elsereturn find(fa[i])…...

53-1 内网代理3 - Netsh端口转发(推荐)

靶场还是用上一篇文章搭建的靶场 :52-5 内网代理2 - LCX端口转发(不推荐使用LCX)-CSDN博客 一、Netsh 实现端口转发 Netsh是Windows自带的命令行脚本工具,可用于配置端口转发。在一个典型的场景中,如果我们位于公网无法直接访问内网的Web服务器,可以利用中间的跳板机通过…...

慧哥充电桩开源平台小程序、PC后、手机商户端历经2年的无数次迭代。

...

四、(1)网络爬虫入门及准备工作(爬虫及数据可视化)

四、&#xff08;1&#xff09;网络爬虫入门及准备工作&#xff08;爬虫及数据可视化&#xff09; 1&#xff0c;网络爬虫入门1.1 百度指数1.2 天眼查1.3 爬虫原理1.4 搜索引擎原理 2&#xff0c;准备工作2.1 分析爬取页面2.2 爬虫拿到的不仅是网页还是网页的源代码2.3 爬虫就是…...

2024华为OD机试真题-分月饼-(C++/Python)-C卷D卷-200分

2024华为OD机试题库-(C卷+D卷)-(JAVA、Python、C++) 题目描述 中秋节,公司分月饼,m 个员工,买了 n 个月饼,m ≤ n,每个员工至少分 1 个月饼,但可以分多个,单人分到最多月饼的个数是 Max1 ,单人分到第二多月饼个数是 Max2 ,Max1 - Max2 ≤ 3 ,单人分到第 n - 1…...

Git 查看提交历史

Git 查看提交历史 Git 是一个强大的版本控制系统&#xff0c;它允许开发人员跟踪代码的变化&#xff0c;并与其他人协作。了解如何查看提交历史对于理解项目的发展和维护代码库至关重要。本文将详细介绍如何使用 Git 查看提交历史&#xff0c;包括不同的命令和选项&#xff0c…...

力扣双指针算法题目:快乐数

目录 1.题目 2.思路解析 3.代码展示 1.题目 . - 力扣&#xff08;LeetCode&#xff09; 2.思路解析 题目意思是将一个正整数上面的每一位拿出来&#xff0c;然后分别求平方&#xff0c;最后将这些数字的平方求和得到一个数字&#xff0c;如此循环&#xff0c;如果在此循环中…...

【Tools】了解人工通用智能 (AGI):未来的智能体

什么是人工通用智能 (AGI)&#xff1f; 人工通用智能&#xff08;Artificial General Intelligence&#xff0c;AGI&#xff09;是指一种能够理解、学习和应用知识&#xff0c;具有像人类一样广泛和通用的认知能力的智能系统。与专门处理特定任务的人工智能&#xff08;AI&…...

华媒舍:8种网站构建推广方法全揭密!

网站构建成为了推广宣传和宣传品牌的关键一环。对于新手&#xff0c;搭建和营销推广网站有可能是一项全新的挑战。下面我们就为大家介绍8种网站搭建和营销推广技巧&#xff0c;帮助你在这些方面取得成功。 1.选择适合自己的网站构建平台选择合适的网站构建平台针对构建一个成功…...

【Scrapy】 深入了解 Scrapy 下载中间件的 process_exception 方法

准我快乐地重饰演某段美丽故事主人 饰演你旧年共寻梦的恋人 再去做没流着情泪的伊人 假装再有从前演过的戏份 重饰演某段美丽故事主人 饰演你旧年共寻梦的恋人 你纵是未明白仍夜深一人 穿起你那无言毛衣当跟你接近 &#x1f3b5; 陈慧娴《傻女》 Scrapy 是…...

无法与IP建立连接,未能下载VSCode服务器

如题&#xff0c;在远程连接服务器的时候突然遇到了这个提示。 查阅了一圈&#xff0c;发现是VSCode版本自动更新惹的祸&#xff01;&#xff01;&#xff01; 在VSCode的帮助->关于这里发现前几天VSCode自动更新了&#xff0c;我的版本号变成了1.100.3 才导致了远程连接出…...

在四层代理中还原真实客户端ngx_stream_realip_module

一、模块原理与价值 PROXY Protocol 回溯 第三方负载均衡&#xff08;如 HAProxy、AWS NLB、阿里 SLB&#xff09;发起上游连接时&#xff0c;将真实客户端 IP/Port 写入 PROXY Protocol v1/v2 头。Stream 层接收到头部后&#xff0c;ngx_stream_realip_module 从中提取原始信息…...

selenium学习实战【Python爬虫】

selenium学习实战【Python爬虫】 文章目录 selenium学习实战【Python爬虫】一、声明二、学习目标三、安装依赖3.1 安装selenium库3.2 安装浏览器驱动3.2.1 查看Edge版本3.2.2 驱动安装 四、代码讲解4.1 配置浏览器4.2 加载更多4.3 寻找内容4.4 完整代码 五、报告文件爬取5.1 提…...

听写流程自动化实践,轻量级教育辅助

随着智能教育工具的发展&#xff0c;越来越多的传统学习方式正在被数字化、自动化所优化。听写作为语文、英语等学科中重要的基础训练形式&#xff0c;也迎来了更高效的解决方案。 这是一款轻量但功能强大的听写辅助工具。它是基于本地词库与可选在线语音引擎构建&#xff0c;…...

Aspose.PDF 限制绕过方案:Java 字节码技术实战分享(仅供学习)

Aspose.PDF 限制绕过方案&#xff1a;Java 字节码技术实战分享&#xff08;仅供学习&#xff09; 一、Aspose.PDF 简介二、说明&#xff08;⚠️仅供学习与研究使用&#xff09;三、技术流程总览四、准备工作1. 下载 Jar 包2. Maven 项目依赖配置 五、字节码修改实现代码&#…...

人工智能--安全大模型训练计划:基于Fine-tuning + LLM Agent

安全大模型训练计划&#xff1a;基于Fine-tuning LLM Agent 1. 构建高质量安全数据集 目标&#xff1a;为安全大模型创建高质量、去偏、符合伦理的训练数据集&#xff0c;涵盖安全相关任务&#xff08;如有害内容检测、隐私保护、道德推理等&#xff09;。 1.1 数据收集 描…...

零知开源——STM32F103RBT6驱动 ICM20948 九轴传感器及 vofa + 上位机可视化教程

STM32F1 本教程使用零知标准板&#xff08;STM32F103RBT6&#xff09;通过I2C驱动ICM20948九轴传感器&#xff0c;实现姿态解算&#xff0c;并通过串口将数据实时发送至VOFA上位机进行3D可视化。代码基于开源库修改优化&#xff0c;适合嵌入式及物联网开发者。在基础驱动上新增…...

LangChain 中的文档加载器(Loader)与文本切分器(Splitter)详解《二》

&#x1f9e0; LangChain 中 TextSplitter 的使用详解&#xff1a;从基础到进阶&#xff08;附代码&#xff09; 一、前言 在处理大规模文本数据时&#xff0c;特别是在构建知识库或进行大模型训练与推理时&#xff0c;文本切分&#xff08;Text Splitting&#xff09; 是一个…...

LUA+Reids实现库存秒杀预扣减 记录流水 以及自己的思考

目录 lua脚本 记录流水 记录流水的作用 流水什么时候删除 我们在做库存扣减的时候&#xff0c;显示基于Lua脚本和Redis实现的预扣减 这样可以在秒杀扣减的时候保证操作的原子性和高效性 lua脚本 // ... 已有代码 ...Overridepublic InventoryResponse decrease(Inventor…...

react菜单,动态绑定点击事件,菜单分离出去单独的js文件,Ant框架

1、菜单文件treeTop.js // 顶部菜单 import { AppstoreOutlined, SettingOutlined } from ant-design/icons; // 定义菜单项数据 const treeTop [{label: Docker管理,key: 1,icon: <AppstoreOutlined />,url:"/docker/index"},{label: 权限管理,key: 2,icon:…...