Elvis 运算符

Elvis 运算符 (?:) 是三元运算符语法的缩写,用于 Groovy 语言。使用 三元运算符语法时,您通常需要重复一个变量两次,如以下 Java 示例所示:

String name = "Elvis Presley";
String displayName = (name != null ? name : "Unknown");

您可以使用 Elvis 运算符(因其与猫王发型相似而得名)。 以下示例展示了如何在 SpEL 表达式中使用 Elvis 运算符:

  • Java

  • Kotlin

ExpressionParser parser = new SpelExpressionParser();

String name = parser.parseExpression("name ?: 'Unknown'").getValue(new Inventor(), String.class);
System.out.println(name);  // 'Unknown'
val parser = SpelExpressionParser()

val name = parser.parseExpression("name ?: 'Unknown'").getValue(Inventor(), String::class.java)
println(name)  // 'Unknown'

SpEL Elvis 运算符还将_空_字符串视作 null 对象。因此, 原始 Java 示例仅接近模拟该运算符的语义:它需要 使用 name != null && !name.isEmpty() 作为谓词才能与 SpEL Elvis 运算符的语义兼容。

自 Spring Framework 7.0 起,SpEL Elvis 运算符支持 java.util.Optional, 并具有透明的解包语义。 例如,给定表达式 A ?: B,如果 Anull 或_空_ Optional, 则表达式评估为 B。但是,如果 A 是非空 Optional,则表达式 评估为 Optional 中包含的对象,从而有效地解包 Optional,这与 A.get() 相关。

以下列表显示了一个更复杂的示例:

  • Java

  • Kotlin

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

Inventor tesla = new Inventor("Nikola Tesla", "Serbian");
String name = parser.parseExpression("name ?: 'Elvis Presley'").getValue(context, tesla, String.class);
System.out.println(name);  // Nikola Tesla

tesla.setName("");
name = parser.parseExpression("name ?: 'Elvis Presley'").getValue(context, tesla, String.class);
System.out.println(name);  // Elvis Presley
val parser = SpelExpressionParser()
val context = SimpleEvaluationContext.forReadOnlyDataBinding().build()

val tesla = Inventor("Nikola Tesla", "Serbian")
var name = parser.parseExpression("name ?: 'Elvis Presley'").getValue(context, tesla, String::class.java)
println(name)  // Nikola Tesla

tesla.setName("")
name = parser.parseExpression("name ?: 'Elvis Presley'").getValue(context, tesla, String::class.java)
println(name)  // Elvis Presley

您可以使用 Elvis 运算符在表达式中应用默认值。以下 示例展示了如何在 @Value 表达式中使用 Elvis 运算符:

@Value("#{systemProperties['pop3.port'] ?: 25}")

这将注入名为 pop3.port 的系统属性的值(如果已定义),或者 如果该属性未定义,则注入 25