spel 多层list嵌套表达式踩坑记

场景

bash 复制代码
Expression exp = spelParser.parseExpression("#{#avgTable?.get(2)?.get(0)}", new TemplateParserContext());
String _result = exp.getValue(evalContext, String.class);

当avgTable?.get(2)为空时,Method threw 'java.lang.IndexOutOfBoundsException' exception.

在 Spring Expression Language (SpEL) 中,安全导航操作符 (?.) 通常可以防止 NullPointerException,但如果它不起作用,可能是以下原因导致的:


1. ?. 无法防止 IndexOutOfBoundsException

问题原因:

  • avgTable?.get(2) 如果 avgTableListsize() <= 2,会抛出 IndexOutOfBoundsException,而不是返回 null
  • ?. 只能防止 NullPointerException(即 avgTablenull 的情况),但无法阻止越界访问。

解决方案:

  • 改用 size() 检查:

    java 复制代码
    "#{(#avgTable != null && #avgTable.size() > 2) ? #avgTable.get(2)?.get(0) : null}"
  • 或使用 SafeList 包装(自定义方法,返回 null 而不是抛异常)。


2. SpEL 版本或解析方式问题

问题原因:

  • 某些旧版本的 SpEL 可能对 ?. 支持不完全。
  • 如果使用的是 TemplateParserContext(如 #{...}),确保语法正确。

解决方案:

  • 升级 Spring 版本(推荐 Spring 5+)。

  • 检查 SpEL 表达式是否正确解析:

    java 复制代码
    Expression exp = parser.parseExpression("#{#avgTable?.get(2)?.get(0)}", new TemplateParserContext());

3. avgTable 不是标准的 Java Bean/List/Map

问题原因:

  • 如果 avgTable 是自定义类,且 get(2) 不是标准方法(如数组访问或特殊数据结构),?. 可能不会按预期工作。
  • 例如,数组访问 arr[2] 越界会直接抛异常,?. 无法阻止。

解决方案:

  • 改用 ListMap 结构。

  • 或手动检查索引:

    java 复制代码
    "#{(#avgTable != null && #avgTable.length > 2) ? #avgTable[2]?.property : null}"

4. 嵌套对象访问方式错误

问题原因:

  • 如果 avgTable.get(2) 返回的对象没有 get(0) 方法,?. 也无法阻止 NoSuchMethodException

  • 例如:

    java 复制代码
    List<String> row = avgTable.get(2);  // 返回 String,但尝试调用 get(0)
    row?.get(0);  // String 没有 get(0) 方法,会抛异常

解决方案:

  • 确保对象结构正确:

    java 复制代码
    "#{#avgTable?.get(2) instanceof List ? #avgTable.get(2)?.get(0) : null}"

5. 表达式解析错误

问题原因:

  • 如果 SpEL 表达式语法错误(如括号不匹配),?. 可能不会生效。

  • 例如:

    java 复制代码
    "#{#avgTable?.get(2)?.get(0}"  // 缺少右括号

解决方案:

  • 检查表达式语法是否正确:

    java 复制代码
    "#{#avgTable?.get(2)?.get(0)}"  // 正确

推荐解决方案

1. 使用 size() 检查索引(适用于 List

java 复制代码
Expression exp = parser.parseExpression(
    "#{(#avgTable != null && #avgTable.size() > 2) ? #avgTable.get(2)?.get(0) : null}",
    new TemplateParserContext()
);

2. 使用 Elvis 运算符提供默认值

java 复制代码
Expression exp = parser.parseExpression(
    "#{#avgTable?.get(2)?.get(0) ?: 'DEFAULT'}",
    new TemplateParserContext()
);

3. 使用 try-catch(不推荐,仅备用)

java 复制代码
Expression exp = parser.parseExpression(
    "#{T(org.springframework.util.ObjectUtils).nullSafeToString(#avgTable?.get(2)?.get(0))}",
    new TemplateParserContext()
);

总结

问题 原因 解决方案
IndexOutOfBoundsException List.get(2) 越界 检查 size()
NullPointerException avgTablenull 使用 ?.
自定义类无 get(0) 对象结构错误 检查类型
SpEL 语法错误 表达式不合法 修复语法
旧版 SpEL 问题 版本兼容性 升级 Spring

如果仍然有问题,建议打印 avgTable 的实际类型和结构:

java 复制代码
System.out.println("avgTable class: " + avgTable.getClass());
System.out.println("avgTable size: " + (avgTable != null ? avgTable.size() : "null"));

这样可以更精准地定位问题。

相关推荐
头发掉光的程序员11 小时前
第七章 利用Direct3D绘制几何体
c++·windows·图形渲染·direct12
许泽宇的技术分享13 小时前
Text2API与Text2SQL深度对比:自然语言驱动的数据交互革命
数据库·windows·microsoft
△曉風殘月〆16 小时前
如何在Windows 10/11家庭版安装组策略编辑器
windows·组策略
野犬寒鸦17 小时前
力扣hot100:搜索二维矩阵与在排序数组中查找元素的第一个和最后一个位置(74,34)
java·数据结构·算法·leetcode·list
啊吧怪不啊吧1 天前
C++之list类的代码及其逻辑详解 (中)
开发语言·数据结构·c++·list
Boxsc_midnight1 天前
【适合中小企业应用的Flask网站部署指南】【小白指南系列】如何在Windows Server服务器上部署Flask网站和SSL证书开启HTTPS
服务器·windows·python·flask
IDOlaoluo1 天前
dotnet-hosting-2.2.6-win.exe 怎么安装?Windows 下完整步骤(附安装包下载)
windows
Bruce_Liuxiaowei1 天前
使用批处理脚本安全清理Windows系统垃圾
网络·windows·安全·网络安全
小陈永不服输1 天前
Windows下RabbitMQ完整安装指南
windows·分布式·rabbitmq
准女婿_2 天前
优考试局域网系统V6.0.0版
linux·windows·用户运营