整理了下最近在项目中使用drools出现的问题,幸好都在开发与测试阶段解决了,未波及到prod。

首先看这样两条规则:

/**
* 规则1_set默认利率a
*/
rule "rate_default_a"
no-loop true
when
$request:AmountRateRequest(calculateEnum == CalculateEnum.INTEREST || calculateEnum == CalculateEnum.AMOUNT_INTEREST)
$response:AmountRateResponse(rateMap not contains LoanTermEnum.LOANTERM3)
$data:DroolsData()
then
DroolsClient.logger.debug("{}执行规则{}", $response.getUserId(), drools.getRule().getName());
$response.getRateMap().put(LoanTermEnum.LOANTERM3, RateFactory.DEFAULT_RATE_A);
update($response);
end /**
* 规则2_set默认利率b
*/
rule "rate_default_b"
no-loop true
when
$request:AmountRateRequest(calculateEnum == CalculateEnum.INTEREST || calculateEnum == CalculateEnum.AMOUNT_INTEREST)
$response:AmountRateResponse(rateMap not contains LoanTermEnum.LOANTERM3)
then
DroolsClient.logger.debug("{}执行规则{}", $response.getUserId(), drools.getRule().getName());
$response.getRateMap().put(LoanTermEnum.LOANTERM3, RateFactory.DEFAULT_RATE_B);
update($response);
end

理想的情况:当规则fire后,rate_default_a规则实行,并set3期利率,得到结果后,由于不满足b规则的when条件(rateMap中3期利率已经存在),则不会继续执行rate_default_b,一切正常,

实际的结果:a执行后触发b、b执行后触发a,造成死循环

原因分析:

  肯定是因为when条件约束失败,导致重复触发,而与规则中修改所相关的就是

$response:AmountRateResponse(rateMap not contains LoanTermEnum.LOANTERM3)

  在drools中,不能通过contains来判断java的map对象是否containsKey。contains 只能用于对象的某个Collection/Array 类型的字段与另外一个值进行比较,作为比较的值可以是一个静态的值,也可以是一个变量(绑定变量或者是一个global 对象),不能操作map。如果需要判断map,建议使用map[keyName]的方式,比如我们这条规则,可以修改为:

$response:AmountRateResponse(rateMap[LoanTermEnum.LOANTERM3] == null)

来做,即可正确判断map的某个值是否为空。

同时对规则中出现的no-loop进行分析:  

  网络中能找到的大部分回答:no-loop属性的作用是用来控制已经执行过的规则在条件再次满足时是否再次执行。默认情况下规则的no-loop属性的值为false,如果no-loop 属性值为true,那么就表示该规则只会被引擎检查一次

  实际效果:no-loop所说的只执行一次,是说当本条规则内如果更新了fact,不会重新触发本条规则的执行。如果像我们上面代码中的情况,a规则和b规则本身都有no-loop true,但a中的udpate仍可以触发b的执行,b也可以触发a。

  如果需要让某条规则只能触发一次,则不能靠no-loop,需要使用lock-on-active true来做。同时注意:虽然规则只能被执行一次是可以做到的,但对于一些场景中,某些规则不希望被触发,并不只是限制次数,还需要结合具体业务来做。

结论和改进:

  1. 不要使用contains操作map,采用map[keyName]的形式

  2. 规则导致的死循环可能有很多种形式,a触发a、ab间相互触发,都可能引起很坏的结果,上线前要谨慎。

  3. 建议在每条规则执行前后增加日志,当出现死循环、或其他不符合预期的结果时能快速定位,方便追踪。

  4. 建议把怀疑有问题的语句拆成最小的单元执行。比如对我们上面的代码稍微修改,只保留一条规则:

/**
* 规则1_set默认利率a
*/
rule "rate_default_a"
no-loop true
when
$request:AmountRateRequest(calculateEnum == CalculateEnum.INTEREST || calculateEnum == CalculateEnum.AMOUNT_INTEREST)
$response:AmountRateResponse(rateMap not contains LoanTermEnum.LOANTERM3)
$data:DroolsData()
then
DroolsClient.logger.debug("{}执行规则{}", $response.getUserId(), drools.getRule().getName());
$response.getRateMap().put(LoanTermEnum.LOANTERM3, RateFactory.DEFAULT_RATE_A);
update($response);
end

  虽然不会导致死循环,但也无法说明到底是因为contains有效、还是no-loop true不触发自身,具体是哪行导致的结果正常。如果我们去掉no-loop true,就会发现依然出现了死循环,发现是contains的问题。

drools规则引擎中易混淆语法分析_相互触发导致死循环分析的更多相关文章

  1. Drools规则引擎-memberOf操作

    场景 规则引擎技术讨论2群(715840230)有同学提出疑问,memberOf的使用过程中如果,memberOf之后的参数不是集合也不是数组,而是格式如"1,2,3,4"的字符串 ...

  2. Drools规则引擎-如果判断某个对象中的集合是否包含指定的值

    规则引擎集合相关处理 在实际生产过程中,有很多关于集合的处理场景,比如一个Fact对象中包含有一个集合,而需要判断该集合是否包含某个值.而Drools规则引擎也提供了多种处理方式,比如通过from.c ...

  3. Drools规则引擎入门指南(一)

    最近项目需要增加风控系统,在经过一番调研以后决定使用Drools规则引擎.因为项目是基于SpringCloud的架构,所以此次学习使用了SpringBoot2.0版本结合Drools7.14.0.Fi ...

  4. drools规则引擎初探

    https://www.cnblogs.com/yuebintse/p/5767996.html 1.drools是什么 Drools是为Java量身定制的基于Charles  Forgy的RETE算 ...

  5. Drools规则引擎

    一.简介 Drools is a Business Rules Management System (BRMS) solution. It provides a core Business Rules ...

  6. Drools 规则引擎环境搭建

    一.关于 drools 规则引擎 前面写过一篇 Drools 规则引擎相关的文章,这篇文章主要记录一下规则引擎的环境搭建和简单示例.不熟悉 drools 的朋友可以看看这篇文章: 自己写个 Drool ...

  7. Spring Boot+Drools规则引擎整合

    目的 官方的Drools范例大都是基于纯Java项目或Maven项目,而基于Spring Boot项目的很少. 本文介绍如何在Spring Boot项目上加上Drools规则引擎. POM依赖 POM ...

  8. Drools规则引擎-判断集合(List)是否包含集合

    问题场景 在使用Drools规则引擎时,有朋友会遇到这样的问题,就是在when部分判断的两个参数都是集合类型,比如两个List,此时要判断一个集合是否包含另外一个集合的内容. 拿一个具体的例子来说明, ...

  9. SpringBoot2 整合 Drools规则引擎,实现高效的业务规则

    本文源码:GitHub·点这里 || GitEE·点这里 一.Drools引擎简介 1.基础简介 Drools是一个基于java的规则引擎,开源的,可以将复杂多变的规则从硬编码中解放出来,以规则脚本的 ...

随机推荐

  1. Error:java: Compilation failed: internal java compiler error 解决办法

    https://blog.csdn.net/jdjdndhj/article/details/70256989

  2. yii2.0 引入autoload.php提示Operation not permitted

    open_basedir()配置下就可以了.比如目录是/www/ad/web/yii/就在/usr/local/nginx/conf/fastcgi.conf里面修改下配置 opendir=/www/ ...

  3. Xamarin.Form与Xamarin.Android或Xamarin.IOS的区别简述

    Xamarin.Form与Xamarin.Android或Xamarin.IOS的区别简述: 可能刚刚接触Xamarin的人来说,对于这个概念比较的模糊,认为这说的不都是同一个东西吗?事实并不是这样的 ...

  4. Java框架spring 学习笔记(七):Spring相关概念

    Spring是开源.轻量级.一站式框架. Spring核心主要两部分 aop:面向切面编程,扩展功能不是修改源代码实现. ioc:控制反转,把对象的创建交给Spring进行配置,比如一个类,在类里面有 ...

  5. Win10系统总是提示"在商店中查找应用"的关闭方法

    Win10系统总是提示"在商店中查找应用"该怎么关闭?win10中打开文件的时候总是提示在商店中查找应用,但是自己的电脑中有程序可以打开这个文件,不需要去商店中下载,该怎么取消这个 ...

  6. “新智认知”杯上海高校程序设计竞赛暨第十七届上海大学程序设计春季联赛(D题,贪心+栈)

    链接:https://ac.nowcoder.com/acm/contest/551/D来源:牛客网 时间限制:C/C++ 1秒,其他语言2秒 空间限制:C/C++ 524288K,其他语言10485 ...

  7. java中mysql查询报错java.sql.SQLException: Before start of result set

    异常:java.sql.SQLException: Before start of result set 解决方法:使用rs.getString();前一定要加上rs.next(); sm = con ...

  8. 从零开始学spring cloud(二) -------- 开始使用Spring Cloud实战微服务

    1.准备工作 2.服务提供者与服务消费者 3.服务发现与服务注册 服务发现: 服务注册表: 服务注册表是一个记录当前可用服务实例的网络信息的数据库,是服务发现机制的核心.服务注册表提供查询API和管理 ...

  9. linux尝试登录失败后锁定用户账户的两种方法

    linux尝试登录失败后锁定用户账户的两种方法 更新时间:2017年06月23日 08:44:31   作者:Carey    我要评论   这篇文章主要给大家分享了linux尝试登录失败后锁定用户账 ...

  10. tiny4412 --Uboot移植(4) 串口

    开发环境:win10 64位 + VMware12 + Ubuntu14.04 32位 工具链:linaro提供的gcc-linaro-6.1.1-2016.08-x86_64_arm-linux-g ...