05.仿简道云公式函数实战-扩展和自定义操作符和函数

2023-12-11 10:44

本文主要是介绍05.仿简道云公式函数实战-扩展和自定义操作符和函数,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1. 前言

在上一篇文章中我们学习了QLExpress的基础操作符和java对象的操作,通过大量的测试用例,我们学习了QLExpress的基础语法与使用,本篇文章,我们介绍使用QLExpress的进阶内容,主要知识点是扩展操作符和自定义操作符。

2. 扩展操作符

需求:实现一个操作符"加",它的功能具备与操作符"+"的功能一样。你是不是想到要用addOperatorWithAlias方法?

实现如下:

/**** 需求:扩展+操作符 使用"加"汉字替代+的功能* @throws Exception*/@Testpublic void ext() throws Exception{ExpressRunner runner = new ExpressRunner(false, true);
//        String expressStr = "20 + 5";String expressStr = "20 加 5";runner.addOperatorWithAlias("加","+","取模操作符定义异常!");// isTraceObject rst = runner.execute(expressStr, null, null, true, true);System.out.println(rst);}

QLExpress是如何实现"加"替换"+"的这个功能呢?首先我们进入addOperatorWithAlias方法的源码

public void addOperatorWithAlias(String aliasName, String name, String errorInfo) throws Exception {if (!this.operatorMap.containsKey(name)) {throw new QLException(name + " 不是系统级别的操作符号,不能设置别名");} else {OperatorBase originalOperator = this.operatorMap.get(name);if (originalOperator == null) {throw new QLException(name + " 不能被设置别名");}OperatorBase destOperator;if (originalOperator instanceof CanClone) {destOperator = ((CanClone)originalOperator).cloneMe(aliasName, errorInfo);} else {Class<OperatorBase> opClass = (Class<OperatorBase>)originalOperator.getClass();Constructor<OperatorBase> constructor;try {constructor = opClass.getConstructor(String.class, String.class, String.class);} catch (Exception e) {throw new QLException(name + " 不能被设置别名:" + e.getMessage());}if (constructor == null) {throw new QLException(name + " 不能被设置别名");}destOperator = constructor.newInstance(aliasName, name, errorInfo);}if (this.operatorMap.containsKey(aliasName)) {throw new RuntimeException("操作符号:\"" + aliasName + "\" 已经存在");}this.addOperator(aliasName, destOperator);}
}

通过这个逻辑我们可知,首先QLExpress根据"+"符号操作符找到OperatorAdd作为originalOperator,然后通过originalOperator.getClass();获取opClass,根据opClass.getConstructor(String.class, String.class, String.class);找到构造函数,根据constructor.newInstance(aliasName, name, errorInfo);反射生成destOperator,最后调用this.addOperator(aliasName, destOperator)等同于addOperator("加",newOperatorAdd("+"));

3.自定义操作符

尽管QLExpress表达式引擎封装了好多操作符,常规的使用没有问题,但是总归项目或者产品上有个性化的操作符。那么这个时候我们应该怎么使用QLExpress来满足需求,这也是我想强调的一个地方就是QLExpress引擎的扩展性很强,我们可以借助QLExpress引擎扩展实现自定义操作符。

首先,我们还是模拟一个需求,比如我们想实现一个操作符union,该操作符功能为所有以参数形式给出的值拼接结果返回。

/**** 自定义操作符 * @throws Exception*/
@Test
public void union() throws Exception{ExpressRunner runner = new ExpressRunner(false, true);String expressStr = "'a' union 'b' union 3";runner.addOperator("union",new OperatorUnion());Object rst = runner.execute(expressStr, null, null, true, true);System.out.println(rst);
}

4. 自定义函数

使用QLExpress如何自定义函数以及QLExpress相关API。在这里我首先给一个简单的列子,通过这个例子我们引出知识点。

需求:实现一个sum函数,通过sum函数计算求和。

/**** 自定义函数* @throws Exception*/
@Test
public void sum() throws Exception{ExpressRunner runner = new ExpressRunner(false, true);String expressStr = "sum(1,2,3)";runner.addFunction("sum",new OperatorSum("sum"));Object rst = runner.execute(expressStr, null, null, true, true);System.out.println(rst);
}

runner.addFunction("sum",new OperatorSum("sum"));我们自定义的函数,要添加到ExpressRunner中,这样在使用sum函数的时候,就能执行我们在OperatorSum这个类中的逻辑。

OperatorSum类的代码就是自定义函数的内容

import com.ql.util.express.Operator;
import com.ql.util.express.OperatorOfNumber;
import com.ql.util.express.exception.QLException;/*** 类描述: 类描述: SUM(number1, [number2], …)函数使所有以参数形式给出的数字相加并返回和。* @author admin* @version 1.0.0* @date 2023/11/20 15:39*/
public class OperatorSum extends Operator {public OperatorSum(String name) {this.name = name;}@Overridepublic Object executeInner(Object[] lists) throws Exception {if (lists.length == 0) {throw new QLException("操作数异常");}Object result = 0;for (int i = 0; i < lists.length; i++) {result = OperatorOfNumber.add(result, lists[i], isPrecise);}return result;}
}

我们在来看QLExpress中addFunction做了什么处理。以下为addFunction方法代码

/*** 添加函数定义** @param name 函数名称* @param op   对应的操作实现类*/
public void addFunction(String name, OperatorBase op) {// 操作符的管理器this.operatorManager.addOperator(name, op);// 语法定义的管理器this.manager.addFunctionName(name);
}

此类位于

语法分析和计算的入口类ExpressRunner中。

咱们暂且先理解到这,就是自定义的函数最后通过addFunction交给了ExpressRunner类中,这样ExpressRunner中调用execute方法就能执行到我们自定义的逻辑。

5.总结

本篇文章主要介绍了使用QLExpress扩展操作符,自定义操作符和自定义函数。从侧面也能反映出来QLExpress的代码扩展性很好。

最近笔者创建了一个圈子纷传

欢迎大家加入一起交流学习。

这篇关于05.仿简道云公式函数实战-扩展和自定义操作符和函数的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/480413

相关文章

PostgreSQL的扩展dict_int应用案例解析

《PostgreSQL的扩展dict_int应用案例解析》dict_int扩展为PostgreSQL提供了专业的整数文本处理能力,特别适合需要精确处理数字内容的搜索场景,本文给大家介绍PostgreS... 目录PostgreSQL的扩展dict_int一、扩展概述二、核心功能三、安装与启用四、字典配置方法

MySQL 中的 CAST 函数详解及常见用法

《MySQL中的CAST函数详解及常见用法》CAST函数是MySQL中用于数据类型转换的重要函数,它允许你将一个值从一种数据类型转换为另一种数据类型,本文给大家介绍MySQL中的CAST... 目录mysql 中的 CAST 函数详解一、基本语法二、支持的数据类型三、常见用法示例1. 字符串转数字2. 数字

Python内置函数之classmethod函数使用详解

《Python内置函数之classmethod函数使用详解》:本文主要介绍Python内置函数之classmethod函数使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地... 目录1. 类方法定义与基本语法2. 类方法 vs 实例方法 vs 静态方法3. 核心特性与用法(1编程客

Python函数作用域示例详解

《Python函数作用域示例详解》本文介绍了Python中的LEGB作用域规则,详细解析了变量查找的四个层级,通过具体代码示例,展示了各层级的变量访问规则和特性,对python函数作用域相关知识感兴趣... 目录一、LEGB 规则二、作用域实例2.1 局部作用域(Local)2.2 闭包作用域(Enclos

从原理到实战深入理解Java 断言assert

《从原理到实战深入理解Java断言assert》本文深入解析Java断言机制,涵盖语法、工作原理、启用方式及与异常的区别,推荐用于开发阶段的条件检查与状态验证,并强调生产环境应使用参数验证工具类替代... 目录深入理解 Java 断言(assert):从原理到实战引言:为什么需要断言?一、断言基础1.1 语

MySQL count()聚合函数详解

《MySQLcount()聚合函数详解》MySQL中的COUNT()函数,它是SQL中最常用的聚合函数之一,用于计算表中符合特定条件的行数,本文给大家介绍MySQLcount()聚合函数,感兴趣的朋... 目录核心功能语法形式重要特性与行为如何选择使用哪种形式?总结深入剖析一下 mysql 中的 COUNT

Java MQTT实战应用

《JavaMQTT实战应用》本文详解MQTT协议,涵盖其发布/订阅机制、低功耗高效特性、三种服务质量等级(QoS0/1/2),以及客户端、代理、主题的核心概念,最后提供Linux部署教程、Sprin... 目录一、MQTT协议二、MQTT优点三、三种服务质量等级四、客户端、代理、主题1. 客户端(Clien

MySQL 中 ROW_NUMBER() 函数最佳实践

《MySQL中ROW_NUMBER()函数最佳实践》MySQL中ROW_NUMBER()函数,作为窗口函数为每行分配唯一连续序号,区别于RANK()和DENSE_RANK(),特别适合分页、去重... 目录mysql 中 ROW_NUMBER() 函数详解一、基础语法二、核心特点三、典型应用场景1. 数据分

在Spring Boot中集成RabbitMQ的实战记录

《在SpringBoot中集成RabbitMQ的实战记录》本文介绍SpringBoot集成RabbitMQ的步骤,涵盖配置连接、消息发送与接收,并对比两种定义Exchange与队列的方式:手动声明(... 目录前言准备工作1. 安装 RabbitMQ2. 消息发送者(Producer)配置1. 创建 Spr

深度解析Spring Boot拦截器Interceptor与过滤器Filter的区别与实战指南

《深度解析SpringBoot拦截器Interceptor与过滤器Filter的区别与实战指南》本文深度解析SpringBoot中拦截器与过滤器的区别,涵盖执行顺序、依赖关系、异常处理等核心差异,并... 目录Spring Boot拦截器(Interceptor)与过滤器(Filter)深度解析:区别、实现