整理了下最近在项目中使用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规则引擎中易混淆语法分析_相互触发导致死循环分析的更多相关文章
-
Drools规则引擎-memberOf操作
场景 规则引擎技术讨论2群(715840230)有同学提出疑问,memberOf的使用过程中如果,memberOf之后的参数不是集合也不是数组,而是格式如"1,2,3,4"的字符串 ...
-
Drools规则引擎-如果判断某个对象中的集合是否包含指定的值
规则引擎集合相关处理 在实际生产过程中,有很多关于集合的处理场景,比如一个Fact对象中包含有一个集合,而需要判断该集合是否包含某个值.而Drools规则引擎也提供了多种处理方式,比如通过from.c ...
-
Drools规则引擎入门指南(一)
最近项目需要增加风控系统,在经过一番调研以后决定使用Drools规则引擎.因为项目是基于SpringCloud的架构,所以此次学习使用了SpringBoot2.0版本结合Drools7.14.0.Fi ...
-
drools规则引擎初探
https://www.cnblogs.com/yuebintse/p/5767996.html 1.drools是什么 Drools是为Java量身定制的基于Charles Forgy的RETE算 ...
-
Drools规则引擎
一.简介 Drools is a Business Rules Management System (BRMS) solution. It provides a core Business Rules ...
-
Drools 规则引擎环境搭建
一.关于 drools 规则引擎 前面写过一篇 Drools 规则引擎相关的文章,这篇文章主要记录一下规则引擎的环境搭建和简单示例.不熟悉 drools 的朋友可以看看这篇文章: 自己写个 Drool ...
-
Spring Boot+Drools规则引擎整合
目的 官方的Drools范例大都是基于纯Java项目或Maven项目,而基于Spring Boot项目的很少. 本文介绍如何在Spring Boot项目上加上Drools规则引擎. POM依赖 POM ...
-
Drools规则引擎-判断集合(List)是否包含集合
问题场景 在使用Drools规则引擎时,有朋友会遇到这样的问题,就是在when部分判断的两个参数都是集合类型,比如两个List,此时要判断一个集合是否包含另外一个集合的内容. 拿一个具体的例子来说明, ...
-
SpringBoot2 整合 Drools规则引擎,实现高效的业务规则
本文源码:GitHub·点这里 || GitEE·点这里 一.Drools引擎简介 1.基础简介 Drools是一个基于java的规则引擎,开源的,可以将复杂多变的规则从硬编码中解放出来,以规则脚本的 ...
随机推荐
-
static 的使用
static用法小结 转自 http://blog.csdn.net/Kendiv/article/details/675941 在C语言中,static的字面意思很容易把我们导入歧途,其实它的作用有 ...
-
openssl与cryptoAPI交互AES加密解密
继上次只有CryptoAPI的加密后,这次要实现openssl的了 动机:利用CryptoAPI制作windows的IE,火狐和chrome加密控件后,这次得加上与android的加密信息交互 先前有 ...
-
poj 3020 Antenna Placement(最小路径覆盖 + 构图)
http://poj.org/problem?id=3020 Antenna Placement Time Limit: 1000MS Memory Limit: 65536K Total Sub ...
-
set_ip_pool
#! /usr/bin/env python# -*- coding: utf-8 -*- import redisimport urllib2class RedisConnect: #clas ...
-
leetcode实现 “10001”+“1011” 返回二进制相加的结果
https://oj.leetcode.com/problems/add-binary/ 实” 1 public class Solution { public String addBinary(St ...
-
Bencode编码解析的C++实现
Ben编码的基本规则 B编码中有4种类型:字符串.整型.列表.字典. 字符串 字符串的编码格式为:<字符串的长度>:<字符串>,其中<>括号中的内容为必需.例如,有 ...
-
Javascript进阶篇——( JavaScript内置对象---下)--Math对象---笔记整理
Math对象使用 Math 的属性和方法: <script type="text/javascript"> var mypi=Math.PI; var myabs=Ma ...
-
sql必知必会(第四版) 学习笔记一
温习一遍简单的sql语法,把自己掌握还不够的地方,做了些笔记.... 1 去重复关键词,distinct select distinct sname from student; 2 限制结果top的用 ...
-
[置顶] linux学习之samba安装问题详解
一.首先查看是否安装samba,命令为:rpm -qa | grep samba 出现如下包表示已经安装,否则没有安装 samba-winbind-clients-3.5.10-125.el6.i68 ...
-
org.springframework.boot.builder.SpringApplicationBuilder.<;init>;
新建了一个Spring cloud项目,启动时报错org.springframework.boot.builder.SpringApplicationBuilder.<init> 翻阅资料 ...