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

相关文章

Python在二进制文件中进行数据搜索的实战指南

《Python在二进制文件中进行数据搜索的实战指南》在二进制文件中搜索特定数据是编程中常见的任务,尤其在日志分析、程序调试和二进制数据处理中尤为重要,下面我们就来看看如何使用Python实现这一功能吧... 目录简介1. 二进制文件搜索概述2. python二进制模式文件读取(rb)2.1 二进制模式与文本

Django调用外部Python程序的完整项目实战

《Django调用外部Python程序的完整项目实战》Django是一个强大的PythonWeb框架,它的设计理念简洁优雅,:本文主要介绍Django调用外部Python程序的完整项目实战,文中通... 目录一、为什么 Django 需要调用外部 python 程序二、三种常见的调用方式方式 1:直接 im

SpringBoot全局异常拦截与自定义错误页面实现过程解读

《SpringBoot全局异常拦截与自定义错误页面实现过程解读》本文介绍了SpringBoot中全局异常拦截与自定义错误页面的实现方法,包括异常的分类、SpringBoot默认异常处理机制、全局异常拦... 目录一、引言二、Spring Boot异常处理基础2.1 异常的分类2.2 Spring Boot默

Mybatis对MySQL if 函数的不支持问题解读

《Mybatis对MySQLif函数的不支持问题解读》接手项目后,为了实现多租户功能,引入了Mybatis-plus,发现之前运行正常的SQL语句报错,原因是Mybatis不支持MySQL的if函... 目录MyBATis对mysql if 函数的不支持问题描述经过查询网上搜索资料找到原因解决方案总结Myb

自定义注解SpringBoot防重复提交AOP方法详解

《自定义注解SpringBoot防重复提交AOP方法详解》该文章描述了一个防止重复提交的流程,通过HttpServletRequest对象获取请求信息,生成唯一标识,使用Redis分布式锁判断请求是否... 目录防重复提交流程引入依赖properties配置自定义注解切面Redis工具类controller

SpringBoot整合 Quartz实现定时推送实战指南

《SpringBoot整合Quartz实现定时推送实战指南》文章介绍了SpringBoot中使用Quartz动态定时任务和任务持久化实现多条不确定结束时间并提前N分钟推送的方案,本文结合实例代码给大... 目录前言一、Quartz 是什么?1、核心定位:解决什么问题?2、Quartz 核心组件二、使用步骤1

SpringBoot整合AOP及使用案例实战

《SpringBoot整合AOP及使用案例实战》本文详细介绍了SpringAOP中的切入点表达式,重点讲解了execution表达式的语法和用法,通过案例实战,展示了AOP的基本使用、结合自定义注解以... 目录一、 引入依赖二、切入点表达式详解三、案例实战1. AOP基本使用2. AOP结合自定义注解3.

Python容器转换与共有函数举例详解

《Python容器转换与共有函数举例详解》Python容器是Python编程语言中非常基础且重要的概念,它们提供了数据的存储和组织方式,下面:本文主要介绍Python容器转换与共有函数的相关资料,... 目录python容器转换与共有函数详解一、容器类型概览二、容器类型转换1. 基本容器转换2. 高级转换示

Spring配置扩展之JavaConfig的使用小结

《Spring配置扩展之JavaConfig的使用小结》JavaConfig是Spring框架中基于纯Java代码的配置方式,用于替代传统的XML配置,通过注解(如@Bean)定义Spring容器的组... 目录JavaConfig 的概念什么是JavaConfig?为什么使用 JavaConfig?Jav

Java 队列Queue从原理到实战指南

《Java队列Queue从原理到实战指南》本文介绍了Java中队列(Queue)的底层实现、常见方法及其区别,通过LinkedList和ArrayDeque的实现,以及循环队列的概念,展示了如何高效... 目录一、队列的认识队列的底层与集合框架常见的队列方法插入元素方法对比(add和offer)移除元素方法