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"));

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

相关推荐
可乐要加冰^-^2 分钟前
云雀文档下载
windows·git·github·石墨文档
caimouse28 分钟前
Reactos 第 9 章 设备驱动 — 9.9 磁盘的设备驱动堆叠
windows·嵌入式硬件
阿维的博客日记1 小时前
Windows自由切换jdk版本
java·windows
山峰哥2 小时前
VBA数据结构之争:Dictionary vs Collection,性能差3倍!
服务器·数据结构·数据库·windows·sql·算法·哈希算法
caimouse11 小时前
Reactos 第 8 章 结构化异常处理 — 8.2 系统空间的结构化异常处理
windows
caimouse12 小时前
Reactos 第 7 章 视窗报文 — 7.3 Win32k 的用户空间回调机制
windows
caimouse12 小时前
Reactos 第 9 章 设备驱动 — 9.5 一组PnP设备驱动模块的实例
网络·windows
神成112 小时前
vmware 上 win7 系统按照 vmware tool
windows
虾壳云官方13 小时前
OpenClaw 2.7.9 Windows 一键部署教程:零基础也能搭建 AI 自动化助手
运维·人工智能·windows·自动化·openclaw·openclaw一键部署
xcLeigh15 小时前
鸿蒙平台 KeePass 密码管理器适配实战:从 Windows 到 鸿蒙PC 的 Electron 迁移指南
windows·electron·web·harmonyos·加密算法·keepass