Drools规则加载变量冲突问题分析
问题现象说明
在个别环境下加载规则时出现:rule/trade/hg/Rule_FY_*.java (53:3948) : Duplicate local variable paraMap,出现此问题的环境为:was8.5+linux2.6,问题的关键现象也在于:开发环境、客户相同的部署环境均未发生过类似异常。
问题分析
经过对drools源码的跟踪、调试,该问题和中间件无关。根本原因是规则脚本中的全局变量和局部变量重复了,由系统在加载文件时的顺序引发此问题:
- 规则脚本中voucher_preprocess.drl中定义了一个全局变量(paraMap),HG-2.0.0.drl等脚本中也定义了paraMap的局部变量,当规则引擎将全局变量作为本规则方法的入参时就会发生冲突。调试出rule/trade/hg/Rule_FY_*.java的源代码,相关代码为:
|
public class Rule_FY_e9cd732bdbd54e998f137a2f40468216 { public static void defaultConsequence(org.drools.spi.KnowledgeHelper drools,**,java.util.Map paraMap ) throws Exception { org.drools.runtime.rule.RuleContext kcontext = drools; Map<String, List<CNSecurityFlow>> originalMap = map.getGroupMap(); ****** Map<String, String> paraMap = new HashMap<String, String>(); ****** } } } |
paraMap变量的冲突点为方法入参和本方法的局部定义变量发生重复定义,其中入参paraMap是规则引擎根据规则的上下文信息自动生成的,其生成的原理为:采用antlr词法分析器对规则脚本进行分析,提取出局部变量,将此局部变量和全局变量列表进行匹配,如在全局变量列表中找到了,则将此加入到方法入参,之所以要将此作为入参,是因为在规则体内可能有用到全局变量的场景。
- 规则引擎何时将全局变量作为入参是有条件的,其中的一个必要条件是在解析规则中的局部变量时其全局变量已经先加载了(和规则文件的加载顺序有关);当voucher_preprocess.drl脚本比HG-2.0.0.drl先加载时便会触发此条件,发生问题,当voucher_preprocess.drl后加载时,在解析HG-2.0.0.drl的相关局部变量时因为未发现有相关的全局变量,在生成方法入参时也就不会生成此全局变量了,也就不会有问题。
- 文件的加载顺序:读取规则文件时,用的是java.io.File.listFiles(),而该方法java官方的说明是:"不保证数组中的字符以特定顺序出现,特别是不保证按字母顺序排序"(绝大部分情况是按字母顺序排序的),原始注释:
|
/** * **** * <p> There is no guarantee that the name strings in the resulting array * will appear in any specific order; they are not, in particular, * guaranteed to appear in alphabetical order. */ public File[] listFiles() { |
结论
对于规则脚本的编写,也应该遵循java的编码规范,比如全局变量的命名以g开头,避免局部变量和全局变量出现同名现象。
可删除voucher_preprocess.drl中的全局变量或对相关规则中的局部变量paraMap进行重命名均可解决此问题。
重现条件
在未报出此问题的环境中,将voucher_preprocess.drl的文件名前添加a(avoucher_preprocess.drl),使其先于HG-2.0.0.drl加载。
附:调试日志
有问题环境规则文件加载顺序(无规则):
|
/home/sofa_home/rules/SJSJG-2.0.0.drl, /home/sofa_home/rules/voucher_qyfhpx.drl, /home/sofa_home/rules/STOCKBUYBACK-2.0.0.drl, /home/sofa_home/rules/voucher_xg.drl, /home/sofa_home/rules/voucher_qygxsf.drl, /home/sofa_home/rules/voucher_etfjj_bp.drl, /home/sofa_home/rules/voucher_zqzh.drl, /home/sofa_home/rules/BJSMX_BFJ-2.0.0.drl, /home/sofa_home/rules/voucher_gpjys.drl, /home/sofa_home/rules/voucher_trade.drl, /home/sofa_home/rules/voucher_xdhgjx.drl, /home/sofa_home/rules/TAJZ-2.0.0.drl, /home/sofa_home/rules/voucher_preprocess.drl, /home/sofa_home/rules/DETAILTOSUMMARY_SH-2.0.0.drl, |
无问题环境规则文件加载顺序(按字母排序):
|
C:\test\sofa_home\rules\BJSJG-2.0.0.drl, C:\test\sofa_home\rules\BJSMX-2.0.0.drl, C:\test\sofa_home\rules\BJSMX_BFJ-2.0.0.drl, C:\test\sofa_home\rules\COMMAND-2.0.0.drl, C:\test\sofa_home\rules\DEALCPQSQFCASH_BJ-2.0.0.drl, C:\test\sofa_home\rules\DEALCPQSQFCASH_GGT_SH-2.0.0.drl, C:\test\sofa_home\rules\DEALCPQSQFCASH_SH-2.0.0.drl, C:\test\sofa_home\rules\DEALCPQSQFCASH_SZ-2.0.0.drl, C:\test\sofa_home\rules\DEALYJQSQFCASH_BJ-2.0.0.drl, C:\test\sofa_home\rules\DEALYJQSQFCASH_GGT_SH-2.0.0.drl, C:\test\sofa_home\rules\DEALYJQSQFCASH_SH-2.0.0.drl, C:\test\sofa_home\rules\DEALYJQSQFCASH_SZ-2.0.0.drl, C:\test\sofa_home\rules\DETAILTOSUMMARY_BJ-2.0.0.drl, C:\test\sofa_home\rules\DETAILTOSUMMARY_GGT_SH-2.0.0.drl, C:\test\sofa_home\rules\DETAILTOSUMMARY_SH-2.0.0.drl, C:\test\sofa_home\rules\DETAILTOSUMMARY_SZ-2.0.0.drl, C:\test\sofa_home\rules\ETFJJCASHSUMMARY-2.0.0.drl, C:\test\sofa_home\rules\FUTURES-2.0.0.drl, C:\test\sofa_home\rules\GP-2.0.0.drl, |
Drools规则加载变量冲突问题分析的更多相关文章
- Spring加载流程源码分析03【refresh】
前面两篇文章分析了super(this)和setConfigLocations(configLocations)的源代码,本文来分析下refresh的源码, Spring加载流程源码分析01[su ...
- Springboot学习04-默认错误页面加载机制源码分析
Springboot学习04-默认错误页面加载机制源码分析 前沿 希望通过本文的学习,对错误页面的加载机制有这更神的理解 正文 1-Springboot错误页面展示 2-Springboot默认错误处 ...
- Springboot 加载配置文件源码分析
Springboot 加载配置文件源码分析 本文的分析是基于springboot 2.2.0.RELEASE. 本篇文章的相关源码位置:https://github.com/wbo112/blogde ...
- Android艺术——Bitmap高效加载和缓存代码分析(2)
Bitmap的加载与缓存代码分析: 图片的压缩 比如有一张1024*768像素的图像要被载入内存,然而最终你要用到的图片大小其实只有128*96,那么我们会浪费很大一部分内存,这显然是没有必要的,下面 ...
- ElasticSearch 启动时加载 Analyzer 源码分析
ElasticSearch 启动时加载 Analyzer 源码分析 本文介绍 ElasticSearch启动时如何创建.加载Analyzer,主要的参考资料是Lucene中关于Analyzer官方文档 ...
- jQuery实现DOM加载方法源码分析
传统的判断dom加载的方法 使用 dom0级 onload事件来进行触发所有浏览器都支持在最初是很流行的写法 我们都熟悉这种写法: window.onload=function(){ ... } 但 ...
- JVM加载类冲突,导致Mybatis查数据库返回NULL的一个小问题
今天碰到个bug,虽然小,但是有点意思 背景是SpringMVC + Mybatis的一个项目,mapper文件里写了一条sql 大概相当于 select a from tableA where b ...
- Linker加载so失败问题分析
WeTest 导读 近期测试反馈一个问题,在旧版本微视基础上覆盖安装新版本的微视APP,首次打开拍摄页录制视频合成时高概率出现crash. 那么我们直奔主题,看看日志: 另外复现的日志中还出现如下信息 ...
- spring启动加载过程源码分析
我们知道启动spring容器两常见的两种方式(其实都是加载spring容器的xml配置文件时启动的): 1.在应用程序下加载 ApplicationContext ctx = new ClassPat ...
随机推荐
- CTest
一.初识CTest CTest是CMake集成的一个测试工具,在使用CMakeLists.txt文件编译工程的时候,CTest会自动configure.build.test和展现测试结果 CTest有 ...
- Hive[1] 初识 及 安装
本文前提是Hadoop & Java & mysql 数据库,已经安装配置好,并且 环境变量均已经配置到位 声明:本笔记参照 学习<Hive 编程指南>而来,如果有错误 ...
- 使用Visual Studio Code开发Asp.Net Core WebApi学习笔记(五)-- Filter
在上一篇里,介绍了中间件的相关内容和使用方法.本篇将介绍Asp.Net Core MVC框架的过滤器的相关内容和使用方法,并简单说明一下与中间件的区别. 第一部分.MVC框架内置过滤器 下图展示了As ...
- 【转】准确理解CSS clear:left/right的含义及实际用途
零.说点什么 好久没更新了.并不是在折腾什么大作,而是广度学习与实践中,加上婚礼等诸多大事,所以产出较少. 今天这篇也只是小作,博客是自己很好的学习工具,只要我学习不止,博客也会不断更新的. 我们平时 ...
- mysql开启全文索引功能
注意:全文检索模块需要mysql开启全文索引功能,开启方法:修改mysql配置文件:window服务器为my.ini,linux服务器为my.cnf,在 [mysqld] 后面加入一行“ft_min_ ...
- 等价表达式 (codevs 1107)题解
[问题描述] 明明进了中学之后,学到了代数表达式.有一天,他碰到一个很麻烦的选择题.这个题目的题干中首先给出了一个代数表达式,然后列出了若干选项,每个选项也是一个代数表达式,题目的要求是判断选项中哪些 ...
- LINUX开启允许对外访问的网络端口
LINUX开启允许对外访问的网络端口 LINUX通过下面的命令可以开启允许对外访问的网络端口: /sbin/iptables -I INPUT -p tcp --dport 8000 -j ACCE ...
- 信号驱动的IO
(1)client1,基于SIGIO的写法: #include <sys/types.h> #include <sys/stat.h> #include <fcntl.h ...
- JQUERY 判断选择器选择的对象 是否存在
判断方法: 直接选择判断,是不正确的方法,因为 $(“#id”) 不管对象是否存在都会返回 object . if($("#id")){ alert('存在'); }else{ a ...
- iOS开发多线程篇—单例模式(ARC)
iOS开发多线程篇—单例模式(ARC) 一.简单说明: 设计模式:多年软件开发,总结出来的一套经验.方法和工具 java中有23种设计模式,在ios中最常用的是单例模式和代理模式. 二.单例模式说明 ...