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

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

相关推荐
Abigail_chow7 分钟前
EXCEL如何快速批量给两字姓名中间加空格
windows·microsoft·excel·学习方法·政务
love530love2 小时前
【笔记】在 MSYS2(MINGW64)中正确安装 Rust
运维·开发语言·人工智能·windows·笔记·python·rust
代码搬运媛3 小时前
“packageManager“: “[email protected]“ 配置如何正确启动项目?
windows·webpack
小道士写程序4 小时前
Qt 5.12 上读取 .xlsx 文件(Windows 平台)
开发语言·windows·qt
异常君9 小时前
Windows 与 Linux 虚拟内存机制对比:设计理念与实现差异
java·linux·windows
搏博11 小时前
将图形可视化工具的 Python 脚本打包为 Windows 应用程序
开发语言·windows·python·matplotlib·数据可视化
电手12 小时前
Win10停更,Win11不好用?现在Mac电脑比Win11电脑更便宜
windows·macos·电脑·mac
拾回程序猿的圈圈∞12 小时前
PyCharm项目和文件运行时使用conda环境的教程
windows·pycharm·conda
波点兔12 小时前
【亲测有效 | Cursor Pro每月500次快速请求扩5倍】(Windows版)Cursor中集成interactive-feedback-mcp
windows·mcp·cursor pro
饮长安千年月14 小时前
JavaSec-SSTI - 模板引擎注入
java·windows·安全·web安全·网络安全·系统安全·安全架构