Spring SpEL在Flink中的应用-SpEL详解

news/2025/1/15 15:10:51/

 

前言

  Spring 表达式语言 Spring Expression Language(简称 SpEL )是一个支持运行时查询和操做对象图的表达式语言 。 语法相似于 EL 表达式 ,但提供了显式方法调用和基本字符串模板函数等额外特性。SpEL 在许多组件中都得到了广泛应用,如 Spring Data、Spring Security、Spring Web Flow 等。它提供了一种非常灵活的方式来查询和操作对象图,从而简化了复杂的业务逻辑和数据操作。本系列文章介绍SpEL与Flink结合应用。


一、SpEL主要功能

  • 属性查询和设置:可以查询和设置对象的属性。例如,#root.name 查询根对象的 "name" 属性。
  • 方法调用:可以调用对象的任意方法。例如,#root.someMethod() 调用根对象的 "someMethod" 方法。
  • 条件表达式:可以使用条件表达式,如 ?>, ?, <, <=, >=, !=, == 来判断值。
  • 布尔和关系运算符:可用于布尔和关系运算。
  • 集合操作:可以用来操作集合,如 size(), contains(), any, all, isEmpty 等。
  • 自定义函数:可以注册自定义函数,并在 SpEL 表达式中调用它们。
  • 类型转换:SpEL 支持类型转换,例如 as 关键字可以用来转换类型。
  • 正则表达式:可以使用正则表达式进行模式匹配。
  • 解析 JSON:SpEL 可以解析 JSON 字符串并查询其内容。

二、POM依赖

首先在 pom.xml 中加入依赖:

<dependency><groupId>org.springframework</groupId><artifactId>spring-expression</artifactId><version>5.2.0.RELEASE</version>
</dependency>

二、SpEL应用

1.解析字面量

    private void evaluateLiteralExpresssions() {Expression exp = parser.parseExpression("'Hello World'");String message = (String) exp.getValue();System.out.println(message);exp = parser.parseExpression("88");Integer value = exp.getValue(Integer.class);System.out.println(value*2);}

2.直接文本上调用方法

示例展示了在字符串上直接调用Java String类的public方法。

private void methodInvocationOnLiterals() {Expression exp = parser.parseExpression("'Hello World'.concat('!')");String message = (String) exp.getValue();println(message);exp = parser.parseExpression("'Hello World'.length()");Integer size = exp.getValue(Integer.class);println(size);exp = parser.parseExpression("'Hello World'.split(' ')[0]");message = (String)exp.getValue();println(message);}

3、访问对象属性和方法

	private void accessingObjectProperties() {User user = new User("John", "Doe",  true, "john.doe@acme.com",30);Expression exp = parser.parseExpression("firstName");println((String)exp.getValue(user));exp = parser.parseExpression("isAdmin()==false");boolean isAdmin = exp.getValue(user, Boolean.class);println(isAdmin);exp = parser.parseExpression("email.split('@')[0]");String emailId = exp.getValue(user, String.class);println(emailId);exp = parser.parseExpression("age");Integer age = exp.getValue(user, Integer.class);println(age);}

4、Json表达式

    public static void jsonExpress(){JSONObject json = new JSONObject();json.put("age",20);StandardEvaluationContext conetxt = new StandardEvaluationContext(json);SpelExpressionParser parser = new SpelExpressionParser();String el="get('age')>10";el="['age']>10";Expression exp = parser.parseExpression(el);Boolean bb=(Boolean)exp.getValue(conetxt);System.out.println(bb);}

5、Flink Row表达式

    public static void rowExpress(){Row row = Row.of("name4", 6000, 104.5d);StandardEvaluationContext conetxt = new StandardEvaluationContext(row);SpelExpressionParser parser = new SpelExpressionParser();String el="getField(2)+getField(2)";Expression exp = parser.parseExpression(el);Object value = exp.getValue(conetxt);Double bb=(Double)value;System.out.println(bb);}

6、调用自定义函数

注册自定义函数,并调用。日期比较代码示例:

    public static void compareDate(){StandardEvaluationContext context = new StandardEvaluationContext(new SpelMethodUtil());context.setVariable("dateOne", new Date());
//        context.setVariable("dateTwo", "2022-01-01");//SpEL ParserExpressionParser parser = new SpelExpressionParser();Expression exp = parser.parseExpression("compareDate(#dateOne, \"2024-01-01\")");Object value = exp.getValue(context);System.out.println(value);}

自定义函数类

public class SpelMethodUtil {public static final String TIMESTAMP_FORMAT = "yyyy-MM-dd HH:mm:ss";public static final String DATE_FORMAT = "yyyy-MM-dd";public static final String TIME_FORMAT = "HH:mm:ss";public static Integer compareDate(Date date, String strDate){Integer result;if(date==null&& StringUtils.isBlank(strDate)){return 0;}else{if(date==null || StringUtils.isBlank(strDate)){return -2;}}String trimDate=strDate.trim();String format = findFormat(trimDate);Date date2 = stringToDate(trimDate, format);result=date.compareTo(date2);return result;}public static Integer compareDate(Date first, Date second){if(first==null&& second==null){return 0;}else{if(first==null || second==null){return -2;}}return first.compareTo(second);}public static Date stringToDate(String dateStr,String format){SimpleDateFormat sdf = new SimpleDateFormat(format);Date date=null;try {date= sdf.parse(dateStr);} catch (ParseException e) {e.printStackTrace();}return date;}/*** 查找与输入的字符型日期相匹配的format* @param strDate* @return*/public static String findFormat(String strDate){String result=null;String trimDate=strDate.trim();int len=trimDate.length();String dateRegex = "";if(len==TIMESTAMP_FORMAT.length()){dateRegex = "^\\d{4}-\\d{2}-\\d{2} \\d{2}:\\d{2}:\\d{2}$";if(trimDate.matches(dateRegex)){result=TIMESTAMP_FORMAT;}}else if(len==DATE_FORMAT.length()){dateRegex = "^\\d{4}-\\d{2}-\\d{2}$";if(trimDate.matches(dateRegex)){result=DATE_FORMAT;}}else if(len==TIME_FORMAT.length()){dateRegex = "^\\d{2}:\\d{2}:\\d{2}$";if(trimDate.matches(dateRegex)){result=TIME_FORMAT;}}else{throw  new RuntimeException("不可识别的日期格式!"+strDate);}return result;}public static Integer addAge(Integer age){return age+4;}
}

七、执行各种操作符(比较、逻辑、算术)

SpEl支持下面几种操作:

关系比较操作:==, !=, <, <=, >, >=
逻辑操作: and, or, not
算术操作: +, -, /, *, %, ^

    private void operators() {User user = new User("John", "Doe", true,"john.doe@acme.com",  30);Expression exp = parser.parseExpression("age > 18");println(exp.getValue(user,Boolean.class));exp = parser.parseExpression("age < 18 and isAdmin()");println(exp.getValue(user,Boolean.class));}

总结

通过示例介绍了SpEl中多种应用示例。大家可以利用这些功能实现更加灵活的功能应用。


http://www.ppmy.cn/news/1329409.html

相关文章

React Native性能优化指南

摘要 本文将介绍在React Native开发中常见的性能优化问题和解决方案&#xff0c;包括ScrollView内无法滑动、热更新导致的文件引用问题、高度获取、强制横屏UI适配、低版本RN适配iOS14、缓存清理、navigation参数取值等。通过代码案例演示和详细说明&#xff0c;帮助开发者更好…

一起玩儿物联网人工智能小车(ESP32)——44. 利用红外测距模块GP2Y0E03实现避障小车

摘要&#xff1a;本文介绍使用红外测距模块GP2Y0E03实现避障小车 在前边已经介绍了两种非接触测距的办法&#xff0c;分别是超声波测距和激光测距&#xff0c;在这里&#xff0c;再介绍另一种常用的测距传感器——红外测距传感器。红外测距的工作原理是&#xff0c;利用红外信号…

洛谷 P1126 机器人搬重物

题目描述 机器人移动学会&#xff08;RMI&#xff09;现在正尝试用机器人搬运物品。机器人的形状是一个直径 1.6 米的球。在试验阶段&#xff0c;机器人被用于在一个储藏室中搬运货物。储藏室是一个 NM 的网格&#xff0c;有些格子为不可移动的障碍。机器人的中心总是在格点上…

java---多线程-02

线程API sleep阻塞 sleep方法处理异常:InterruptedException. 当一个线程调用sleep方法处于睡眠阻塞的过程中,该线程的interrupt()方法被调用时,sleep方法会抛出该异常从而打断睡眠阻塞. package thread;/*** sleep方法要求必须处理中断异常:InterruptedException* 当一个线程调…

LeetCode.2765. 最长交替子数组

题目 2765. 最长交替子数组 分析 为了得到数组 nums 中的最长交替子数组的长度&#xff0c;需要分别计算以每个下标结尾的最长交替子数组的长度。为了方便处理&#xff0c;计算过程中需要考虑长度等于 1 的最长交替子数组&#xff0c;再返回结果时判断最长交替子数组的长度…

复杂高层建筑环境多模态导航服务和引导管理机器人系统设计(预告)

课题基础 机器人工程ROS方向应用型本科毕业设计重点课题学生验收成果 将上面这篇所涉及的算法等应用到如下环境中。 Gazebo新环境AWS RoboMaker Hospital医院场景适用于ROS1和ROS2 高层可以简化为多层测试。最典型的就是两层及以上。 简介 随着城市化进程的加速和高层建筑…

i18n多国语言Internationalization的动态实现

一、数据动态的更新 在上一篇i18n多国语言Internationalization的实现-CSDN博客&#xff0c;可能会遇到一个问题&#xff0c;我们在进行英文或中文切换时&#xff0c;并没有办法对当前的数据进行动态的更新。指的是什么意思呢&#xff1f;当前app.js当中一个组件内容&#xff…

【算法】用JAVA代码实现LRU 【缓存】【LRU】

LRU(Least Recently Used)是一种常见的缓存淘汰策略,用于在缓存空间不足时确定哪些数据应该被淘汰。其基本原则是淘汰最近最少被访问的数据。 工作原理: 最近使用优先: LRU算法基于这样的思想:最近被使用的数据很可能在短时间内还会被使用,因此保留这些数据有助于提高缓…