`
linliangyi2007
  • 浏览: 1015136 次
  • 性别: Icon_minigender_1
  • 来自: 福州
社区版块
存档分类
最新评论

山寨版的简易表达式解析器

阅读更多
山寨表达式解析器的背景:

    这里说说咖啡偶为啥突然“发神经”,要自己做一个表达式解析器。首先声明,偶不是一个喜欢自己造轮子的家伙。在Java这个每天都出新东西的社区中造轮子是一件很无聊,也很没有成就感的事,嘎嘎!

    开始的时候,我们的想法就是用jBPM来实现基于SOA的跨系统的工作流(ESB的残缺版)。完成技术预演后,就发现jBPM有点不够用,主要在于:1.Handler处理模式很不错,但还是不够灵活,需要编写java代码。2.没有web图形化的界面。 3.无法支持“中国特色”的流程配制(做过OA的都知道,国内的用户需求有多么的BT)。于是乎,我们开始了对jBPM的山寨过程。

    在这个过程中,比较关键的一块就是需要一个非常简单的表达式解析器。简单到可以让工程实施人员,甚至最终用户能够用的那种。(jBPM自带的BeanScript是基于Java语言的,对用户使用还是有一定要求的)。于是就有了现在的山寨版的表达式引擎。

    我们自己做完以后,用了一下,感觉虽然功能非常的简单,但确实能很大程度的增加jBPM的Handler的灵活性。再一看,这个引擎其实跟jBPM和工作流没有必然联系的,完全可以用在其他需要动态执行脚本的环境中。咖啡想了想,貌似开源社区木有类似的东东(也许偶孤陋寡闻哈,有的话,请大家告知一声,我也学习一下),就觉得可以拿出来分享一下,也许能给大家一定的帮助。最重要的是,大家可以根据自己需要修改扩充哦~~~~
(PS: IK Expression 2.0 已经发布,请访问http://linliangyi2007.iteye.com/blog/337069


用途
  • jBPM的判断节点DicisionHanlder,根据表达式判读选取路径
  • jBPM的任务分配AssignHandler,根据表达式返回任务授权人
  • jBPM的Action及Event,实现简单任务,如邮件通知
  • 其他的java应用,需要简单的动态表达式的地方



表达式语法规范

1. 支持的参数类型:
a) 数字型 :
  • i. 整形 integer   : -2321 , 34234
  • ii. 长整型 long    :3245235235L
  • iii. 单精度浮点 float : 342.555F
  • iv. 双精度浮点 double: 234234.3423
b) 字符型:“a-zA-Z012456789”
c) 布尔型:true、false
d) 日期时间型:[2008-08-08] 或 [2009-01-01 12:33:14]
e) 扩展类型:List对象集合 (PS: 该类型仅作为运算结果返回,不能做输入参数)

2. 分割符:
a) 括号 "("  ")" —— 标识优先级
b) 逗号   "," —— 分隔函数的参数
c) 方括号 "["  "]" —— 标识日期型常量
d) 双引号 """ —— 标识字符型常量
e) 美元号   "$"   —— 函数前缀
f) 转义符   "\" —— 字符串转义,支持\\ , \”, \r , \n , \t

3. 支持的运算符:




4. 支持的内部函数:
内置函数是目前解析器已经实现的一些非常简单、实用的函数



功能例子说明
1. 支持+、- 、* 、/ 、%(取模) 常规的算术运算,支持括号优先级
如:常见的OA中用于年休假工资计算公式
3000 / 21.5 *(12 - 转正月份)/ 2   

其中,“转正月份”可以是上下文变量

2. 支持字符串相加 , 如:
“ABC”+(123+10) => “ABC133”;
“ABC”+ 123 + 10 = > “ABC12310”


3. 支持 > >= < <= == !=逻辑比较运算,返回布尔值
    3-1.支持数值大小比较 :
1234 > 223  —— true 

    3-2.支持字符大小比较 :
 “1234” > “223”—— fasle

    3-3.支持日期大小比较 :
 [2008-12-23] >= [2008-08-08] —— true


3-4支持各类型常量和变量同null的 == 与 != 比较 ,如:
 申请人!=null  

(其中,“申请人”为执行上下文的变量)

4. 支持逻辑与、逻辑或、逻辑非运算, 如:
true && $DAYEQUALS([2008-01-01] , [2008-11-01]) ——false
true || $DAYEQUALS([2008-01-01] , [2008-11-01]) —— true
true && !$DAYEQUALS([2008-01-01] , [2008-11-01]) —— true


5. 支持结果集合拼接 , 如:
1000 / 10  #  [2008-12-23] > $SYSDATE() # “ABC” + 123 
结果为包含 100 , false , “ABC123”三种不同类型对象的List

6. 支持函数与操作符混合、嵌套调用,如:
!$DAYEQUALS($CALCDATE($SYSDATE(),0,0 , (8+11-5*(6/3)) * (2- 59 % 7),0 ,0,0 ) , [2008-10-01])  —— true


引擎API的QuickStart
实际上,如果只是使用IKExpression,那么只要掌握两Class和两个方法,是不是超级的简单!

类org.wltea.expression.ExpressionEvaluator
方法1:
public static Object evaluate(String expression, Collection<Variable> variables)

说明:传入表达式和表达式上下文的变量,执行表达式返回结果
参数1 :String expression, 要传入执行的表达式
参数2 :Collection<Variable> variables 表达式上下文的变量集合(详细请看类org.wltea.expression.datameta.Variable的说明)。
返回值:表达式执行结果,可能是以下类型的java对象中的一种。
Int、long、float、double、boolean、String、Date、List
方法用途:执行表达式

方法2:
public static String compileExpression(String expression) throws IllegalExpressionException

说明:传入表达式,编译并检查表达式,返回表达式内部的逆波兰型式字窜
参数1 :String expression, 要传入执行的表达式
异常:IllegalExpressionException 表达式异常,如方法的参数类型不正确,或者将日期型与布尔型做相等比较等,都会抛出表达式异常。
返回值:表达式的逆波兰式表示,且所有运算符将使用内部形式代替,例如,表达式
“8+11-5*(6/3)”
将变成
“8 11 PLUS 5 6 3 DIV MUTI MINUS“

方法用途:预编译表达式,检查表达式是否有语法错误

类org.wltea.expression.datameta.Variable

该类是用来表示表达式的上下文变量的,咖啡在上面的例子中用到了大量的上下文变量,这是也是表达式最有用的地方,比如:报销审批流程中,判断流程走向的表达式,(申请金额 > 10000) : “总经理审批”,需要“申请金额“变量;计算年休假津贴的表达式,3000 / 21.5 *(12 - 转正月份)/ 2需要“转正月份” 变量。这些变量通过evaluate(String expression, Collection<Variable> variables)方法中的variables参数传入表达式中。而Variable类型变量的构造十分的简单,它是标准的POJO。

方法1 : 使用构造函数new Variable()生成Variable
public Variable(String variableName , DataType variableDataType , Object variableValue) 

参数1——变量名,如设置为“转正月份”“申请金额”,变量名可以是中文,也可以是英文,使用中文的好处是使表达式容易阅读,使得最终用户也能编写。
参数2——变量类型, 它是DataType枚举类,可以用的有
		//字符窜	DATATYPE_STRING ,
		//布尔类	DATATYPE_BOOLEAN ,
		//整数		DATATYPE_INT ,
		//长整数	DATATYPE_LONG ,
		//浮点数	DATATYPE_FLOAT ,
		//双精度浮点	DATATYPE_DOUBLE ,
		//日期时间	DATATYPE_DATE

对于集合对象DATATYPE_COLLECTION型,目前只支持结果输出,暂不支持变量的输入(可能考虑后期加入)。另外DATATYPE_NULL类型是解析器内部类型,也不对外开放,如果设置变量类型为DATATYPE_NULL型,将抛出异常。
参数3——变量值。对于变量值,唯一要注意的是,如果你的类型是“日期时间DATATYPE_DATE”,那么变量值的字符串要遵循"yyyy-MM-dd HH:mm:ss"格式

方法2: 使用静态方法public static Variable createVariable(String variableName, DataType variableDataType , String variableValueStr) 生成Variable。以上静态方法的参数同构造函数一样,请参考方法1部分参数说明。

      关于这个山寨版的表达式解析器,咖啡先谈到这里,希望这个东东对大家有用。实际上,解析器还支持非常方便的函数扩展的,举例说,你可以自定义一个 $邮件(邮件地址 , 标题, 内容)的函数,那么,你就可以用它在表达式中发送简单的邮件提醒了,特别适用于jBPM的节点任务和事件来调用,嘎嘎!
      如果大家感兴趣的话,咖啡将继续第二部分《简易表达式解析器 之 大家一起来山寨》,计划详细说明如何扩展改造解析器功能,同时谈谈相关的函数引擎算法实现。


附件为IKExpression的jar包,以及完整的Eclipse开发环境源码(有很详细的注释哦~)和Java API DOC




分享到:
评论
10 楼 linliangyi2007 2009-02-11  
midea0978 写道

问题:变量名称不能包含数字,例如
4S店价 * 1.2


兄弟,变量是不能字母打头的啊,你可以用 _4S店价。
这个很多语言都不允许吧。
9 楼 midea0978 2009-02-11  
问题:变量名称不能包含数字,例如
4S店价 * 1.2
8 楼 linliangyi2007 2009-02-11  
兄弟们,预告一下,2.0版本准备出炉啦,采用了优化过的全新算法。届时会放到google code上开源。会有更详细的文档,更强的功能。

现在进入测试和文档阶段了,预估最快在3月初能发布
7 楼 whtinj2ee 2008-12-30  
简单的才是最好的,够用就好,非常感谢
6 楼 linliangyi2007 2008-12-27  
dearshor 写道

  linliangyi2007,建议把你这个东西也放到Google Code(或其他开放协同开发平台)上,并不断完善,做大做强。相信国人一样能做出优秀的开源项目来。也许若干年以后,我们完全能在project里用上国人开发的开源项目作为infrastructure &amp; utilities了。


感谢大牛们的支持,不过弱弱的说一句,俺还没用过google code的东东,对开源开发的运作方式也是不了解的,所以有点不得其门而入的无奈。

我先找人问问
5 楼 dearshor 2008-12-26  
  linliangyi2007,建议把你这个东西也放到Google Code(或其他开放协同开发平台)上,并不断完善,做大做强。
相信国人一样能做出优秀的开源项目来。也许若干年以后,我们完全能在project里用上国人开发的开源项目作为infrastructure & utilities了。
4 楼 linliangyi2007 2008-12-25  
jindw 写道

呵呵,赞,我的JSEL和这个有几分类似。


哈哈,还找到了通道中的高人 !JSEL走得更远些,向你学习,多多指教啊:D 。
3 楼 jindw 2008-12-25  
呵呵,赞,我的JSEL和这个有几分类似。
2 楼 linliangyi2007 2008-12-25  
superscorpio 写道

不知道mvel能不能满足你的要求。赞啊赞!


3Q,去look一下mvel为何方圣神
1 楼 superscorpio 2008-12-24  
不知道mvel能不能满足你的要求。

赞啊赞!

相关推荐

    VB控制计算机并口示例(含完整可以运行源代码)

    VB控制计算机并口示例(含完整可以运行源代码) 可以通过并口直接控制MCU,做SW控制不错,关键还可以学习并口硬件控制学习。含详细源代码哦

    python毕业设计基于PyTorch的手语识别系统源码+数据集(完整项目代码)

    python毕业设计基于PyTorch的手语识别系统源码+数据集(完整项目代码),本资源中的源码都是经过本地编译过可运行的,评审分达到98分,资源项目的难度比较适中,内容都是经过助教老师审定过的能够满足学习、毕业设计、期末大作业和课程设计使用需求,如果有需要的话可以放心下载使用。 python毕业设计基于PyTorch的手语识别系统源码+数据集(完整项目代码)python毕业设计基于PyTorch的手语识别系统源码+数据集(完整项目代码)python毕业设计基于PyTorch的手语识别系统源码+数据集(完整项目代码)python毕业设计基于PyTorch的手语识别系统源码+数据集(完整项目代码)python毕业设计基于PyTorch的手语识别系统源码+数据集(完整项目代码)python毕业设计基于PyTorch的手语识别系统源码+数据集(完整项目代码)python毕业设计基于PyTorch的手语识别系统源码+数据集(完整项目代码)python毕业设计基于PyTorch的手语识别系统源码+数据集(完整项目代码)python毕业设计基于PyTorch的手语识别系统源码+数据集(完整项目代

    基于Unet的树种分别识别模型

    基于Unet的树种分别识别模型

    精选毕设项目-富文本解析,折线图,MD5,bluebird.zip

    精选毕设项目-富文本解析,折线图,MD5,bluebird

    图书管理系统(基于ASP .NET)

    《图书管理系统(基于ASP .NET)》是一款专为学习者设计的应用程序,旨在提供一个全面的图书管理平台。系统的设计采用ASP .NET技术,这是一款由微软开发的用于构建动态网站、web应用和web服务的强大工具。ASP .NET框架以其高效、安全和易于维护的特点,深受开发者的喜爱。 该系统包含了多个核心模块,这些模块覆盖了图书管理的主要功能。有图书录入模块,它允许管理员录入图书的基本信息,如书名、作者、出版社、ISBN号、分类等。图书查询模块提供给用户方便快捷的搜索功能,用户可以根据书名、作者、关键词等条件进行检索。此外,借阅与归还模块确保图书的流通管理,记录图书的借阅状态,提醒用户按时归还,并处理超期罚款等事务。 系统还具备用户管理模块,允许用户注册、登录、修改个人信息。对于权限管理,后台有专门的管理员角色,他们可以对用户进行操作,如分配权限、冻结或解冻账户。同时,系统的统计分析模块能够生成各类报表,如图书借阅量、热门书籍、用户活跃度等,这些数据对于图书馆运营决策有着重要参考价值。 在。内容来源于网络分享,如有侵权请联系我删除。另外如果没有积分的同学需要下载,请私信我。

    精选毕设项目-查拼音.zip

    精选毕设项目-查拼音

    精选毕设项目-音乐在线歌词搜索.zip

    精选毕设项目-音乐在线歌词搜索

    思维导图制作-会计初级知识重难点-会计务实-所有者权益

    本专刊的主要目的是帮助初学者系统化和结构化地掌握会计知识。我们采用思维导图的形式,将复杂的会计概念和流程进行有效的简化,旨在让学习者能够更清晰地理解这些内容,并增强记忆效果。通过视觉化的方式,读者不仅能够感受到会计知识的关联性,还能轻松掌握关键点,提升学习效率。无论是在学习新知识还是复习旧知识时,这种方法都能够为学习者提供极大的便利和帮助。

    配网两阶段鲁棒优化调度模型 关键词:两阶段鲁棒优化,CCG算法,储能 仿真算例采用33节点,采用matlab+yalmip+cplex编写,两阶段模型采用CCG算法求解 模型中一阶段变量主要包括01

    配网两阶段鲁棒优化调度模型 关键词:两阶段鲁棒优化,CCG算法,储能 仿真算例采用33节点,采用matlab+yalmip+cplex编写,两阶段模型采用CCG算法求解。 模型中一阶段变量主要包括01变量和无功优化变量,核心变量主要存在于二阶段,因此在叠加二阶段变量优化过程中更容易得到最优解,所以有限次迭代即得到收敛的结果。 模型以网损为目标,包括功率平衡、网络潮流、电压电流、蓄电池出力以及无功设备出力等约束。 复现《两阶段鲁棒优化的主动配电网动态无功优化》-熊壮壮,具体内容可自行下载了解。

    1..1行列式的定义.ppt

    1..1行列式的定义.ppt

    精选毕设项目-地图定位.zip

    精选毕设项目-地图定位

    MMC整流器平均值模型simulink仿真,19电平,采用交流电流内环,直流电压外环控制,双二阶广义积分器锁相环,PI解耦环流抑制器,调制方式为最近电平逼近调制,完美运行 波形一二为直流侧电压电流

    MMC整流器平均值模型simulink仿真,19电平,采用交流电流内环,直流电压外环控制,双二阶广义积分器锁相环,PI解耦环流抑制器,调制方式为最近电平逼近调制,完美运行。 波形一二为直流侧电压电流,波形三四分别为主控制器及环流抑制器输出调制信号。

    疫苗发布和接种预约系统-springboot毕业项目,适合计算机毕-设、实训项目、大作业学习.zip

    Spring Boot是Spring框架的一个模块,它简化了基于Spring应用程序的创建和部署过程。Spring Boot提供了快速启动Spring应用程序的能力,通过自动配置、微服务支持和独立运行的特性,使得开发者能够专注于业务逻辑,而不是配置细节。Spring Boot的核心思想是约定优于配置,它通过自动配置机制,根据项目中添加的依赖自动配置Spring应用。这大大减少了配置文件的编写,提高了开发效率。Spring Boot还支持嵌入式服务器,如Tomcat、Jetty和Undertow,使得开发者无需部署WAR文件到外部服务器即可运行Spring应用。 Java是一种广泛使用的高级编程语言,由Sun Microsystems公司(现为Oracle公司的一部分)在1995年首次发布。Java以其“编写一次,到处运行”(WORA)的特性而闻名,这一特性得益于Java虚拟机(JVM)的使用,它允许Java程序在任何安装了相应JVM的平台上运行,而无需重新编译。Java语言设计之初就是为了跨平台,同时具备面向对象、并发、安全和健壮性等特点。 Java语言广泛应用于企业级应用、移动应用、桌面应用、游戏开发、云计算和物联网等领域。它的语法结构清晰,易于学习和使用,同时提供了丰富的API库,支持多种编程范式,包括面向对象、命令式、函数式和并发编程。Java的强类型系统和自动内存管理减少了程序错误和内存泄漏的风险。随着Java的不断更新和发展,它已经成为一个成熟的生态系统,拥有庞大的开发者社区和持续的技术创新。Java 8引入了Lambda表达式,进一步简化了并发编程和函数式编程的实现。Java 9及以后的版本继续在模块化、性能和安全性方面进行改进,确保Java语言能够适应不断变化的技术需求和市场趋势。 MySQL是一个关系型数据库管理系统(RDBMS),它基于结构化查询语言(SQL)来管理和存储数据。MySQL由瑞典MySQL AB公司开发,并于2008年被Sun Microsystems收购,随后在2010年,Oracle公司收购了Sun Microsystems,从而获得了MySQL的所有权。MySQL以其高性能、可靠性和易用性而闻名,它提供了多种特性来满足不同规模应用程序的需求。作为一个开源解决方案,MySQL拥有一个活跃的社区,不断为其发展和改进做出贡献。它的多线程功能允许同时处理多个查询,而其优化器则可以高效地执行复杂的查询操作。 随着互联网和Web应用的快速发展,MySQL已成为许多开发者和公司的首选数据库之一。它的可扩展性和灵活性使其能够处理从小规模应用到大规模企业级应用的各种需求。通过各种存储引擎,MySQL能够适应不同的数据存储和检索需求,从而为用户提供了高度的定制性和性能优化的可能性。

    jQuery实现左右切换全屏轮播图特效源码.zip

    这是一种全屏轮播风格的特效,使用HTML、CSS和Javript编写。轮播图包含多张图片和对应的文本介绍,通过自动滑动和手动切换两种方式,展示出不同的内容。该轮播图在网页头部或者特定板块上使用,能够为用户提供直观的视觉体验和丰富的内容呈现。而且,该轮播图可以灵活地设置大小、位置、动画等属性,便于根据实际需求进行个性化定制。

    精选毕设项目-图片预览带后端.zip

    精选毕设项目-图片预览带后端

    精选毕设项目-番茄时钟.zip

    精选毕设项目-番茄时钟

    精选毕设项目-简单的商城小应用.zip

    精选毕设项目-简单的商城小应用

    精选毕设项目-仿zcool站酷.zip

    精选毕设项目-仿zcool站酷

    精选毕设项目-录音机.zip

    精选毕设项目-录音机

    南京理工大学毕业论文overleaf LaTex模板,微调版

    南京理工大学毕业论文overleaf LaTex模板,按照我个人的写作需求修改后的版本

Global site tag (gtag.js) - Google Analytics