使用不同的根对象类型调用相同的 Spring EL 表达式失败

Calling same Spring EL expressions with different root object types fails

让我们考虑以下测试:

@Test
public void testSameExpressionDifferentRootObjectClass() {
    SpelParserConfiguration config = new SpelParserConfiguration(SpelCompilerMode.IMMEDIATE, Thread.currentThread().getContextClassLoader());
    SpelExpressionParser parser = new SpelExpressionParser(config);
    StandardEvaluationContext context = new StandardEvaluationContext();
    Expression expr = parser.parseExpression("'Test: ' + #root");

    assertThat(expr.getValue(context, 42L)).isEqualTo("Test: 42");
    assertThat(expr.getValue(context, "string")).isEqualTo("Test: string");
    assertThat(expr.getValue(context, 42L)).isEqualTo("Test: 42");
    assertThat(expr.getValue(context, "string")).isEqualTo("Test: string");
}

第三个断言失败,但出现以下异常:

org.springframework.expression.spel.SpelEvaluationException: EL1072E: An exception occurred whilst evaluating a compiled expression

at org.springframework.expression.spel.standard.SpelExpression.getValue(SpelExpression.java:328) at org.example.ExpressionTest.testSameExpressionDifferentRootObjectClass(ExpressionProvidersTest.java:36)

Caused by: java.lang.ClassCastException: java.lang.Long cannot be cast to java.lang.String at spel.Ex2.getValue(Unknown Source) at org.springframework.expression.spel.standard.SpelExpression.getValue(SpelExpression.java:318) ... 31 more

这出乎我的意料,在文档中找不到任何关于此约束的内容。我是不是做错了什么?

对于这种情况,您应该使用 MIXED 模式;查看 javadocs...

public enum SpelCompilerMode {

    /**
     * The compiler is switched off; this is the default.
     */
    OFF,

    /**
     * In immediate mode, expressions are compiled as soon as possible (usually after 1 interpreted run).
     * If a compiled expression fails it will throw an exception to the caller.
     */
    IMMEDIATE,

    /**
     * In mixed mode, expression evaluation silently switches between interpreted and compiled over time.
     * After a number of runs the expression gets compiled. If it later fails (possibly due to inferred
     * type information changing) then that will be caught internally and the system switches back to
     * interpreted mode. It may subsequently compile it again later.
     */
    MIXED

}