跳到主要内容

函数

DeepSeek V3 中英对照 Functions

你可以通过注册用户定义的函数来扩展 SpEL,这些函数可以在表达式中使用 #functionName(…​) 语法进行调用。与标准方法调用类似,函数调用也支持可变参数

函数可以通过 setVariable() 方法在 EvaluationContext 实现中注册为变量

提示

StandardEvaluationContext 还定义了 registerFunction(…​) 方法,这些方法提供了一种便捷的方式来将函数注册为 java.lang.reflect.Methodjava.lang.invoke.MethodHandle

注意

由于函数在求值上下文中与变量共享同一个命名空间,因此必须注意确保函数名称和变量名称不会发生冲突。

以下示例展示了如何注册一个用户定义的函数,以便通过 java.lang.reflect.Method 使用反射来调用:

Method method = ...;

EvaluationContext context = SimpleEvaluationContext.forReadOnlyDataBinding().build();
context.setVariable("myFunction", method);
java

例如,考虑以下反转字符串的工具方法:

public abstract class StringUtils {

public static String reverseString(String input) {
return new StringBuilder(input).reverse().toString();
}
}
java

你可以注册并使用前面的方法,如下例所示:

ExpressionParser parser = new SpelExpressionParser();

EvaluationContext context = SimpleEvaluationContext.forReadOnlyDataBinding().build();
context.setVariable("reverseString",
StringUtils.class.getMethod("reverseString", String.class));

// evaluates to "olleh"
String helloWorldReversed = parser.parseExpression(
"#reverseString('hello')").getValue(context, String.class);
java

一个函数也可以注册为 java.lang.invoke.MethodHandle。如果 MethodHandle 的目标和参数在注册之前已经完全绑定,这种方式可能会带来更高效的用例;不过,部分绑定的句柄也是支持的。

考虑 String#formatted(Object…​) 实例方法,它根据模板和可变数量的参数(varargs)生成消息。

你可以将 formatted 方法注册并作为 MethodHandle 使用,如下例所示:

ExpressionParser parser = new SpelExpressionParser();
EvaluationContext context = SimpleEvaluationContext.forReadOnlyDataBinding().build();

MethodHandle mh = MethodHandles.lookup().findVirtual(String.class, "formatted",
MethodType.methodType(String.class, Object[].class));
context.setVariable("message", mh);

// evaluates to "Simple message: <Hello World>"
String message = parser.parseExpression("#message('Simple message: <%s>', 'Hello World', 'ignored')")
.getValue(context, String.class);
java

如上所述,绑定 MethodHandle 并注册绑定的 MethodHandle 也是支持的。如果目标和所有参数都已绑定,这种方式可能会更高效。在这种情况下,SpEL 表达式中不需要任何参数,如下例所示:

ExpressionParser parser = new SpelExpressionParser();
EvaluationContext context = SimpleEvaluationContext.forReadOnlyDataBinding().build();

String template = "This is a %s message with %s words: <%s>";
Object varargs = new Object[] { "prerecorded", 3, "Oh Hello World!", "ignored" };
MethodHandle mh = MethodHandles.lookup().findVirtual(String.class, "formatted",
MethodType.methodType(String.class, Object[].class))
.bindTo(template)
// Here we have to provide the arguments in a single array binding:
.bindTo(varargs);
context.setVariable("message", mh);

// evaluates to "This is a prerecorded message with 3 words: <Oh Hello World!>"
String message = parser.parseExpression("#message()")
.getValue(context, String.class);
java