跳到主要内容
版本:7.0.3

运算符

Hunyuan 7b 中英对照 Operators

Spring表达式语言支持以下几种运算符:

关系运算符

关系运算符(等于、不等于、小于、小于或等于、大于、大于或等于)通过使用标准运算符符号来支持。这些运算符既适用于Number类型,也适用于实现了Comparable接口的类型。以下列表展示了一些关系运算符的示例:

// evaluates to true
boolean trueValue = parser.parseExpression("2 == 2").getValue(Boolean.class);

// evaluates to false
boolean falseValue = parser.parseExpression("2 < -5.0").getValue(Boolean.class);

// evaluates to true
boolean trueValue = parser.parseExpression("'black' < 'block'").getValue(Boolean.class);

// uses CustomValue:::compareTo
boolean trueValue = parser.parseExpression("new CustomValue(1) < new CustomValue(2)").getValue(Boolean.class);
备注

null 进行大于或小于比较时,遵循一个简单的规则:null 被视为“无”(即不等于零)。因此,任何其他值总是大于 nullX > null 总是 true),而没有任何其他值会小于“无”(X < null 总是 false)。

如果你更喜欢数值比较,建议避免基于数字的 null 比较,而是使用与零的比较(例如,X > 0X < 0)。

每个符号运算符也可以指定为纯文本等价形式。这样可以避免在表达式所嵌入的文档类型(例如XML文档)中,所使用的符号具有特殊含义时出现的问题。这些文本等价形式如下:

  • lt (<)

  • gt (>)

  • le (<=)

  • ge (>=)

  • eq (==)

  • ne (!=)

所有文本操作符都不区分大小写。

除了标准的关系运算符外,SpEL还支持betweeninstanceof以及基于正则表达式的matches运算符。以下列表展示了这三种运算符的示例:

boolean result;

// evaluates to true
result = parser.parseExpression(
"1 between {1, 5}").getValue(Boolean.class);

// evaluates to false
result = parser.parseExpression(
"1 between {10, 15}").getValue(Boolean.class);

// evaluates to true
result = parser.parseExpression(
"'elephant' between {'aardvark', 'zebra'}").getValue(Boolean.class);

// evaluates to false
result = parser.parseExpression(
"'elephant' between {'aardvark', 'cobra'}").getValue(Boolean.class);

// evaluates to true
result = parser.parseExpression(
"123 instanceof T(Integer)").getValue(Boolean.class);

// evaluates to false
result = parser.parseExpression(
"'xyz' instanceof T(Integer)").getValue(Boolean.class);

// evaluates to true
result = parser.parseExpression(
"'5.00' matches '^-?\\d+(\\.\\d{2})?$'").getValue(Boolean.class);

// evaluates to false
result = parser.parseExpression(
"'5.0067' matches '^-?\\d+(\\.\\d{2})?$'").getValue(Boolean.class);
备注

between 运算符的语法是 <input> between {<range_begin>, <range_end>>,这实际上是 <input> >= <rangeBegin> && <input> <= <range_end> 的快捷方式。

因此,1 between {1, 5} 的结果是 true,而 1 between {5, 1} 的结果是 false

备注

在使用基本类型时需要小心,因为它们会立即被包装成相应的包装类型。例如,1 instanceof T(int) 的结果是 false,而 1 instanceof T(Integer) 的结果是 true

逻辑运算符

SpEL支持以下逻辑(布尔)运算符:

  • and&&

  • or||

  • not!

所有的文本操作符都不区分大小写。

以下示例展示了如何使用逻辑运算符:

// -- AND --

// evaluates to false
boolean falseValue = parser.parseExpression("true and false").getValue(Boolean.class);

// evaluates to true
String expression = "isMember('Nikola Tesla') and isMember('Mihajlo Pupin')";
boolean trueValue = parser.parseExpression(expression).getValue(societyContext, Boolean.class);

// -- OR --

// evaluates to true
boolean trueValue = parser.parseExpression("true or false").getValue(Boolean.class);

// evaluates to true
String expression = "isMember('Nikola Tesla') or isMember('Albert Einstein')";
boolean trueValue = parser.parseExpression(expression).getValue(societyContext, Boolean.class);

// -- NOT --

// evaluates to false
boolean falseValue = parser.parseExpression("!true").getValue(Boolean.class);

// -- AND and NOT --

String expression = "isMember('Nikola Tesla') and !isMember('Mihajlo Pupin')";
boolean falseValue = parser.parseExpression(expression).getValue(societyContext, Boolean.class);

字符串运算符

您可以在字符串上使用以下运算符。

  • 连接(+

  • 减法(-

    • 用于包含单个字符的字符串
  • 重复(*

以下示例展示了String操作符的用法:

// -- Concatenation --

// evaluates to "hello world"
String helloWorld = parser.parseExpression("'hello' + ' ' + 'world'")
.getValue(String.class);

// -- Character Subtraction --

// evaluates to 'a'
char ch = parser.parseExpression("'d' - 3")
.getValue(char.class);

// -- Repeat --

// evaluates to "abcabc"
String repeated = parser.parseExpression("'abc' * 2")
.getValue(String.class);

数学运算符

你可以对数字使用以下运算符,并且会遵循标准的运算符优先级规则。

  • 加法(+

  • 减法(-

  • 自增(++

  • 自减(--

  • 乘法(*

  • 除法(/

  • 取模(%

  • 指数幂(^

除法和取模运算符也可以用纯文本的形式来表示。这样可以避免在表达式所嵌入的文档类型(例如XML文档)中,所使用的符号具有特殊含义时出现的问题。这些纯文本形式的等价表达如下:

  • div (/)
  • mod (%)

所有的文本操作符都不区分大小写。

备注

递增和递减运算符可以与前缀表示法(++A--A)或后缀表示法(A++A--)一起使用,适用于可以写入数据的变量或属性。

以下示例展示了数学运算符的使用:

Inventor inventor = new Inventor();
EvaluationContext context = SimpleEvaluationContext.forReadWriteDataBinding().build();

// -- Addition --

int two = parser.parseExpression("1 + 1").getValue(int.class); // 2

// -- Subtraction --

int four = parser.parseExpression("1 - -3").getValue(int.class); // 4

double d = parser.parseExpression("1000.00 - 1e4").getValue(double.class); // -9000

// -- Increment --

// The counter property in Inventor has an initial value of 0.

// evaluates to 2; counter is now 1
two = parser.parseExpression("counter++ + 2").getValue(context, inventor, int.class);

// evaluates to 5; counter is now 2
int five = parser.parseExpression("3 + ++counter").getValue(context, inventor, int.class);

// -- Decrement --

// The counter property in Inventor has a value of 2.

// evaluates to 6; counter is now 1
int six = parser.parseExpression("counter-- + 4").getValue(context, inventor, int.class);

// evaluates to 5; counter is now 0
five = parser.parseExpression("5 + --counter").getValue(context, inventor, int.class);

// -- Multiplication --

six = parser.parseExpression("-2 * -3").getValue(int.class); // 6

double twentyFour = parser.parseExpression("2.0 * 3e0 * 4").getValue(double.class); // 24.0

// -- Division --

int minusTwo = parser.parseExpression("6 / -3").getValue(int.class); // -2

double one = parser.parseExpression("8.0 / 4e0 / 2").getValue(double.class); // 1.0

// -- Modulus --

int three = parser.parseExpression("7 % 4").getValue(int.class); // 3

int oneInt = parser.parseExpression("8 / 5 % 2").getValue(int.class); // 1

// -- Exponential power --

int maxInt = parser.parseExpression("(2^31) - 1").getValue(int.class); // Integer.MAX_VALUE

int minInt = parser.parseExpression("-2^31").getValue(int.class); // Integer.MIN_VALUE

// -- Operator precedence --

int minusTwentyOne = parser.parseExpression("1+2-3*8").getValue(int.class); // -21

赋值运算符

要设置一个属性,使用赋值运算符(=)。这通常在调用setValue时完成,但也可以在调用getValue时完成。以下示例展示了使用赋值运算符的两种方法:

Inventor inventor = new Inventor();
EvaluationContext context = SimpleEvaluationContext.forReadWriteDataBinding().build();

parser.parseExpression("name").setValue(context, inventor, "Aleksandar Seovic");

// alternatively
String aleks = parser.parseExpression(
"name = 'Aleksandar Seovic'").getValue(context, inventor, String.class);

重载运算符

默认情况下,SpEL的Operation枚举中定义的数学运算(ADDSUBTRACTDIVIDEMULTIPLYMODULUSPOWER)支持数字等简单类型。通过提供OperatorOverloader的实现,表达式语言可以支持对这些其他类型的运算。

例如,如果我们想重载ADD运算符,以便可以使用+号来连接两个列表,我们可以按照以下方式实现一个自定义的OperatorOverloader

pubic class ListConcatenation implements OperatorOverloader {

@Override
public boolean overridesOperation(Operation operation, Object left, Object right) {
return (operation == Operation.ADD &&
left instanceof List && right instanceof List);
}

@Override
public Object operate(Operation operation, Object left, Object right) {
if (operation == Operation.ADD &&
left instanceof List list1 && right instanceof List list2) {

List result = new ArrayList(list1);
result.addAll(list2);
return result;
}
throw new UnsupportedOperationException(
"No overload for operation %s and operands [%s] and [%s]"
.formatted(operation, left, right));
}
}

如果我们将ListConcatenation注册为StandardEvaluationContext中的OperatorOverloader,那么我们就可以像下面的例子所示那样计算表达式,例如{1, 2, 3} + {4, 5}

StandardEvaluationContext context = new StandardEvaluationContext();
context.setOperatorOverloader(new ListConcatenation());

// evaluates to a new list: [1, 2, 3, 4, 5]
parser.parseExpression("{1, 2, 3} + {2 + 2, 5}").getValue(context, List.class);
备注

OperatorOverloader不会改变运算符的默认语义。例如,上面例子中的2 + 2仍然会计算出4

备注

任何使用重载运算符的表达式都无法编译。详情请参见编译器限制