欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

SpringBoot SpEL語法掃盲與查詢手冊的實現(xiàn)

 更新時間:2020年05月21日 14:36:57   作者:小灰灰Blog  
這篇文章主要介紹了SpringBoot SpEL語法掃盲與查詢手冊的實現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧

Spring 表達(dá)式語言簡稱為 SpEL,一種類似 Ognl 的對象圖導(dǎo)航語言(對于 ognl 不熟悉的同學(xué)可以參考一下: Ognl 系列博文)

SeEL 為 Spring 提供了豐富的想象空間,除了一些基本的表達(dá)式操作之外,還支持

  • 訪問 bean 對象
  • 調(diào)用方法,訪問(修改)類(對象)屬性
  • 計算表達(dá)式
  • 正則匹配
  • ...

 I. 語法百科

以下內(nèi)容均來自官方文檔: https://docs.spring.io/spring-framework/docs/5.2.1.RELEASE/spring-framework-reference/core.html#expressions

1. 字面表達(dá)式

Spel 支持strings, numeric values (int, real, hex), boolean, and null等基本類型,實例如下

ExpressionParser parser = new SpelExpressionParser();

// evals to "Hello World"
String helloWorld = (String) parser.parseExpression("'Hello World'").getValue();

// double 類型
double avogadrosNumber = (Double) parser.parseExpression("6.0221415E+23").getValue();

// evals to 2147483647
int maxValue = (Integer) parser.parseExpression("0x7FFFFFFF").getValue();

boolean trueValue = (Boolean) parser.parseExpression("true").getValue();

Object nullValue = parser.parseExpression("null").getValue();

請注意,字符串需要用單引號包括,浮點數(shù)默認(rèn)為 double 類型,用null表示null object

輸出結(jié)果

str: Hello World
double: 6.0221415E23
int: 2147483647
bool: true
null: null

2. Inline List

通過{}來表明 List 表達(dá)式,一個空的列表直接用{}表示

ExpressionParser parser = new SpelExpressionParser();
// Integer列表
List numbers = (List) parser.parseExpression("{1,2,3,4}").getValue();
System.out.println("list: " + numbers);

// List的元素為List
List<List> listlOfLists = (List) parser.parseExpression("{{'a','b'},{'x','y'}}").getValue();
System.out.println("List<List> : " + listlOfLists);

輸出結(jié)果

list: [1, 2, 3, 4]
List<List> : [[a, b], [x, y]]

3. Inline map

{key:value}來表示 map 表達(dá)式,空 Map 直接用{:}表示

private void map() {
  ExpressionParser parser = new SpelExpressionParser();
  Map map = (Map) parser.parseExpression("{txt:'Nikola',dob:'10-July-1856'}").getValue();
  System.out.println("map: " + map);
  Map mapOfMaps =
      (Map) parser.parseExpression("{txt:{first:'Nikola',last:'Tesla'},dob:{day:10,month:'July',year:1856}}")
          .getValue();
  System.out.println("Map<Map>: " + mapOfMaps);
}

輸出結(jié)果

map: {txt=Nikola, dob=10-July-1856}
Map<Map>: {txt={first=Nikola, last=Tesla}, dob={day=10, month=July, year=1856}}

4. 數(shù)組

數(shù)組可以借助new構(gòu)造方法來實現(xiàn),通過下標(biāo)ary[index]的方式訪問數(shù)組中的元素

private void array() {
   ExpressionParser parser = new SpelExpressionParser();
   int[] numbers1 = (int[]) parser.parseExpression("new int[4]").getValue();
   System.out.println("array: " + JSON.toJSONString(numbers1));

   // Array with initializer
   int[] numbers2 = (int[]) parser.parseExpression("new int[]{1,2,3}").getValue();
   System.out.println("array: " + JSON.toJSONString(numbers2));

   // Multi dimensional array
   int[][] numbers3 = (int[][]) parser.parseExpression("new int[4][5]").getValue();
   System.out.println("array: " + JSON.toJSONString(numbers3));


   int[] nums = new int[]{1, 3, 5};
   EvaluationContext context = new StandardEvaluationContext();
   context.setVariable("num", nums);

   // 通過下標(biāo)訪問數(shù)組中的元素
   Integer numVal = parser.parseExpression("#num[1]").getValue(context, Integer.class);
   System.out.println("numVal in array: " + numVal);
}

輸出如下

array: [0,0,0,0]
array: [1,2,3]
array: [[0,0,0,0,0],[0,0,0,0,0],[0,0,0,0,0],[0,0,0,0,0]]
numVal in array: 3

5. 表達(dá)式

Spel 支持一些 Java 語法中常規(guī)的比較判斷,算數(shù)運算,三元表達(dá)式,類型判斷,matches正則匹配等基表表達(dá)式

下面給出一些簡單的實例

public void expression() {
  ExpressionParser parser = new SpelExpressionParser();
  // 運算
  System.out.println("1+2= " + parser.parseExpression("1+2").getValue());
  // 比較
  System.out.println("1<2= " + parser.parseExpression("1<2").getValue());
  System.out.println("true ? hello : false > " + parser.parseExpression("3 > 2 ? 'hello': 'false' ").getValue());
  // instanceof 判斷,請注意靜態(tài)類,用T進(jìn)行包裝
  System.out.println("instance : " + parser.parseExpression("'a' instanceof T(String)").getValue());
  //正則表達(dá)式
  System.out.println("22 是否為兩位數(shù)字 :" + parser.parseExpression("22 matches '\\d{2}'").getValue());
}

輸出結(jié)果

1+2= 3
1<2= true
true ? hello : false > hello
instance : true
22 是否為兩位數(shù)字 :true

6. Type 與靜態(tài)類

如果想獲取 Class 對象,或者訪問靜態(tài)成員/方法,可以借助T()語法來實現(xiàn)

比如我們有一個靜態(tài)類

public static class StaClz {
  public static String txt = "靜態(tài)屬性";

  public static String hello(String tag) {
    return txt + " : " + tag;
  }
}

如果希望訪問靜態(tài)屬性txt, 表達(dá)式可以寫成T(com.git.hui.boot.spel.demo.BasicSpelDemo.StaClz).txt,請注意圓括號中的是完整簽名;訪問靜態(tài)方法方式類似

public void type() {
  // class,靜態(tài)類
  ExpressionParser parser = new SpelExpressionParser();
  String name =
      parser.parseExpression("T(com.git.hui.boot.spel.demo.BasicSpelDemo.StaClz).txt").getValue(String.class);
  System.out.println("txt: " + name);

  String methodReturn =
      parser.parseExpression("T(com.git.hui.boot.spel.demo.BasicSpelDemo.StaClz).hello" + "('一灰灰blog')")
          .getValue(String.class);
  System.out.println("static method return: " + methodReturn);

  // class類獲取
  Class stringClass = parser.parseExpression("T(String)").getValue(Class.class);
  System.out.println("class: " + stringClass.getName());
}

輸出結(jié)果如下

txt: 靜態(tài)屬性
static method return: 靜態(tài)屬性 : 一灰灰blog
class: java.lang.String

上面的寫法,請重點看一下T(String),這里的 String 沒有用完整的包路徑,即直接位于java.lang包下的類,是可以省略掉完整包名的,就像我們平時寫代碼時,也不需要顯示的加一個import java.lang.*

7. 構(gòu)造方法

上面介紹 array 的時候,就介紹了使用new來創(chuàng)建數(shù)組對象,當(dāng)然也可以直接構(gòu)造其他的普通對象, 如我們新建一個測試類

public static class Person {
  private String name;

  private int age;

  public Person(String name, int age) {
    this.name = name;
    this.age = age;
  }

  public String getName() {
    return name;
  }

  public int getAge() {
    return age;
  }

  @Override
  public String toString() {
    return "Person{" + "txt='" + name + '\'' + ", age=" + age + '}';
  }
}

通過 SpEl 創(chuàng)建一個對象的實例

public void construct() {
  ExpressionParser parser = new SpelExpressionParser();
  Person person = parser.parseExpression("new com.git.hui.boot.spel.demo.BasicSpelDemo.Person('一灰灰', 20)")
      .getValue(Person.class);
  System.out.println("person: " + person);
}

輸出結(jié)果如下:

person: Person{txt='一灰灰', age=20}

請注意,構(gòu)造方法中類的完整簽名

8. 變量引用

細(xì)心的小伙伴,在上面介紹數(shù)組的成員演示的實例中,寫法如"#num[1]",這個 num 前面有一個#,這是一個語法定義,有#修飾的表示變量訪問

要理解這一小節(jié),首先得理解EvaluationContext, 在我們的 SpEL 表達(dá)式的解析中,getValue有一個參數(shù)就是這個 Context,你可以將他簡單理解為包含一些對象的上下文,我們可以通過 SpEL 的語法,來訪問操作 Context 中的某些成員、成員方法屬性等

一般的操作過程如下:

  • context.setVariable("person", person); 向EvaluationContext中塞入成員變量
  • parser.parseExpression(xxx).getValue(context) 解析 SpEL 表達(dá)式,context 必須作為傳參丟進(jìn)去哦

一個簡單的實例

public void variable() {
  ExpressionParser parser = new SpelExpressionParser();
  Person person = new Person("一灰灰blog", 18);
  EvaluationContext context = new StandardEvaluationContext();
  context.setVariable("person", person);

  String name = parser.parseExpression("#person.getName()").getValue(context, String.class);
  System.out.println("variable name: " + name);

  Integer age = parser.parseExpression("#person.age").getValue(context, Integer.class);
  System.out.println("variable age: " + age);
}

輸出結(jié)果如下

variable name: 一灰灰blog
variable age: 18

友情提示,如果訪問對象的私有 Field/method,會拋異常

9. 函數(shù)

Context 中的變量,除了是我們常見的基本類型,普通的對象之外,還可以是方法,在setVariable時,設(shè)置的成員類型為method即可

public void function() {
  try {
    ExpressionParser parser = new SpelExpressionParser();
    EvaluationContext context = SimpleEvaluationContext.forReadOnlyDataBinding().build();
    // 注冊一個方法變量,參數(shù)為method類型
    context.setVariable("hello", StaClz.class.getDeclaredMethod("hello", String.class));

    String ans = parser.parseExpression("#hello('一灰灰')").getValue(context, String.class);
    System.out.println("function call: " + ans);
  } catch (Exception e) {
    e.printStackTrace();
  }
}

輸出結(jié)果如下

function call: 靜態(tài)屬性 : 一灰灰

10. bean 訪問

在 Spring 中,什么對象最常見?當(dāng)然是 bean, 那么我們可以直接通過 SpEL 訪問 bean 的屬性、調(diào)用方法么?

要訪問 bean 對象,所以我們的EvaluationContext中需要包含 bean 對象才行

借助BeanResolver來實現(xiàn),如context.setBeanResolver(new BeanFactoryResolver(applicationContext));
其次訪問 bean 的前綴修飾為@符號
為了演示這種場景,首先創(chuàng)建一個普通的 Bean 對象

@Data
@Component
public class BeanDemo {

  private String blog = "https://spring.hhui.top";

  private Integer num = 8;

  public String hello(String name) {
    return "hello " + name + ", welcome to my blog " + blog + ", now person: " + num;
  }
}

接著我們需要獲取ApplicationContext,所以可以稍微改一下我們的測試類,讓它繼承自ApplicationContextAware

private ApplicationContext applicationContext;

@Override
public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
  this.applicationContext = applicationContext;
}


public void bean() {
  ExpressionParser parser = new SpelExpressionParser();
  StandardEvaluationContext context = new StandardEvaluationContext();
  context.setBeanResolver(new BeanFactoryResolver(applicationContext));

  // 獲取bean對象
  BeanDemo beanDemo = parser.parseExpression("@beanDemo").getValue(context, BeanDemo.class);
  System.out.println("bean: " + beanDemo);

  // 訪問bean方法
  String ans = parser.parseExpression("@beanDemo.hello('一灰灰blog')").getValue(context, String.class);
  System.out.println("bean method return: " + ans);
}

上面的寫法和之前的并沒有太大的區(qū)別,實際輸出結(jié)果如下

bean: BeanDemo(blog=https://spring.hhui.top, num=8)
bean method return: hello 一灰灰blog, welcome to my blog  https://spring.hhui.top, now person: 8

11. ifElse

SpEL 支持三元表達(dá)式,在上述的表達(dá)式中也給出了實例

public void ifThenElse() {
  // 三元表達(dá)式,? :
  ExpressionParser parser = new SpelExpressionParser();
  String ans = parser.parseExpression("true ? '正確': '錯誤'").getValue(String.class);
  System.out.println("ifTheElse: " + ans);
}

輸出結(jié)果如下

ifTheElse: 正確

12. elvis

xx != null ? xx : yy => xx?:yy

這個也屬于我們經(jīng)常遇到的一種場景,如果 xx 為 null,則返回 yy;否則直接返回 xx;簡化寫法為 elvis 寫法: xx?:yy

public void elvis() {
  // xx != null ? xx : yy => xx?:yy
  ExpressionParser parser = new SpelExpressionParser();
  EvaluationContext context = new StandardEvaluationContext();
  context.setVariable("name", null);
  String name = parser.parseExpression("#name?:'Unknown'").getValue(context, String.class);
  System.out.println("elvis-before " + name);

  context.setVariable("name", "Exists!");
  name = parser.parseExpression("#name?:'Unknown'").getValue(context, String.class);
  System.out.println("elvis-after " + name);
}

輸出結(jié)果如下

elvis-before Unknown
elvis-after Exists!

13. 安全表達(dá)式

在 java 中,最常見最討厭的是一個就是 NPE 的問題,SpEL 中當(dāng)然也可能出現(xiàn)這種情況,但是若在 SpEL 中進(jìn)行非空判斷,那就很不優(yōu)雅了,SpEL 提供了xx?.yy的寫法來避免 npe,即

xx == null ? null : xx.yy => xx?.yy

舉例說明

public void safeOperate() {
  // 防npe寫法, xx == null ? null : xx.yy => xx?.yy
  ExpressionParser parser = new SpelExpressionParser();
  Person person = new Person(null, 18);

  String name = parser.parseExpression("name?.length()").getValue(person, String.class);
  System.out.println("safeOperate-before: " + name);

  person.name = "一灰灰blog";
  name = parser.parseExpression("name?.length()").getValue(person, String.class);
  System.out.println("safeOperate-after: " + name);
}

輸出結(jié)果如下

safeOperate-before: null
safeOperate-after: 7

14. 容器截取

遍歷容器,獲取子集,相當(dāng)于 jdk8 Stream 中 filter 用法,語法格式如下

xx.?[expression], 請注意中括弧中的表達(dá)式必須返回 boolean

舉例說明

public void collectionSelection() {
  // 容器截取,返回滿足條件的子集
  // xx.?[expression] , 將滿足expression的子元素保留,返回一個新的集合,類似容器的 filter
  List<Integer> list = new ArrayList<>(Arrays.asList(1, 3, 4, 6, 7, 8, 9));
  ExpressionParser parser = new SpelExpressionParser();

  EvaluationContext context = SimpleEvaluationContext.forReadOnlyDataBinding().build();
  context.setVariable("list", list);
  // 用 #this 來指代列表中的迭代元素
  List<Integer> subList = (List<Integer>) parser.parseExpression("#list.?[#this>5]").getValue(context);
  System.out.println("subList: " + subList);


  Map<String, Integer> map = new HashMap<>();
  map.put("a", 1);
  map.put("b", 10);
  map.put("c", 4);
  map.put("d", 7);
  context.setVariable("map", map);
  // 表達(dá)式內(nèi)部用key, value 來指代map的k,v
  Map subMap = parser.parseExpression("#map.?[value < 5]").getValue(context, Map.class);
  System.out.println("subMap: " + subMap);

  subMap = parser.parseExpression("#map.?[key == 'a']").getValue(context, Map.class);
  System.out.println("subMap: " + subMap);
}

輸出結(jié)果如下

subList: [6, 7, 8, 9]
subMap: {a=1, c=4}
subMap: {a=1}

注意

  • 在列表表達(dá)式中,可以通過#this來指代列表中的每一個元素
  • 在 map 表達(dá)式中,通過key, value來分別指代 map 中的k,v

15. 容器映射

將一個集合通過某種規(guī)則,映射為另一種集合,相當(dāng)于 jdk8 Stream 中的 map 用法,語法如下

xx.![expression], 將表達(dá)式計算的結(jié)果作為輸出容器中的成員

舉例如下

public void collectionProjection() {
  // 容器操作之后,生成另一個容器, 類似lambda中的map方法
  // xx.![expression]

  List<Integer> list = new ArrayList<>(Arrays.asList(1, 3, 4, 6, 7, 8, 9));
  ExpressionParser parser = new SpelExpressionParser();
  EvaluationContext context = SimpleEvaluationContext.forReadOnlyDataBinding().build();
  context.setVariable("list", list);

  // 用 #this 來指代列表中的迭代元素
  List newList = parser.parseExpression("#list.![#this * 2]").getValue(context, List.class);
  System.out.println("newList: " + newList);


  Map<String, Integer> map = new HashMap<>();
  map.put("a", 1);
  map.put("b", 10);
  map.put("c", 4);
  map.put("d", 7);
  context.setVariable("map", map);
  List newListByMap = parser.parseExpression("#map.![value * 2]").getValue(context, List.class);
  System.out.println("newListByMap: " + newListByMap);
}

輸出結(jié)果如下:

newList: [2, 6, 8, 12, 14, 16, 18]
newListByMap: [2, 20, 8, 14]

16. 表達(dá)式模板

SpEL 還提供了一種自定義表達(dá)式模板的方式,將字面量和表達(dá)式放在一起使用,比如下面這一條語句

"random number is #{T(java.lang.Math).random()}"

其中#{T(java.lang.Math).random()}是一個 SpEL 表達(dá)式,左邊的是普通字符串,這種寫法也常見于@Value注解中的屬性寫法,當(dāng)然直接通過上面的寫法執(zhí)行這個語句會報錯,這個時候需要指定ParserContext

舉例說明

public void template() {
  // 模板,混合字面文本與表達(dá)式,使用 #{} 將表達(dá)式包裹起來
  ExpressionParser parser = new SpelExpressionParser();
  String randomPhrase = parser.parseExpression("random number is #{T(java.lang.Math).random()}",
      ParserContext.TEMPLATE_EXPRESSION).getValue(String.class);
  System.out.println("template: " + randomPhrase);
}

輸出結(jié)果如下

template: random number is 0.10438946298113871

17. 小結(jié)

SpEL 屬于非常強大的表達(dá)式語言了,就我個人的感覺而言,它和 OGNL 有些像,當(dāng)它們的上下文中包含了 Spring 的上下文時,可以訪問任何的 bean,而你可以借助它們的語法規(guī)范,做各種事情

推薦我之前的一個項目,https://github.com/liuyueyi/quick-fix,利用 ognl 結(jié)合ApplicationContext,可以隨心所欲的訪問控制應(yīng)用中的任何 bean 對象

II. 其他

0. 項目

工程:https://github.com/liuyueyi/spring-boot-demo

源碼:https://github.com/liuyueyi/spring-boot-demo/spring-boot/013-spel

以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。

相關(guān)文章

  • Java中Stream流對多個字段進(jìn)行排序的方法

    Java中Stream流對多個字段進(jìn)行排序的方法

    我們在處理數(shù)據(jù)的時候經(jīng)常會需要進(jìn)行排序后再返回給前端調(diào)用,比如按照時間升序排序,前端展示數(shù)據(jù)就是按時間先后進(jìn)行排序,下面這篇文章主要給大家介紹了關(guān)于Java中Stream流對多個字段進(jìn)行排序的相關(guān)資料,需要的朋友可以參考下
    2023-10-10
  • 解決Java?properties文件里面如何寫"\"的問題

    解決Java?properties文件里面如何寫"\"的問題

    由于properties使用“\”相當(dāng)于是java的轉(zhuǎn)義符,如果想要寫出\的效果,只需修改相應(yīng)的寫法即可,對java?properties文件里的"\"寫法感興趣的朋友一起看看吧
    2022-04-04
  • eclipse中自動生成構(gòu)造函數(shù)的兩種方法

    eclipse中自動生成構(gòu)造函數(shù)的兩種方法

    下面小編就為大家?guī)硪黄猠clipse中自動生成構(gòu)造函數(shù)的兩種方法。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-10-10
  • Jackson常用方法以及jacksonUtil工具類詳解

    Jackson常用方法以及jacksonUtil工具類詳解

    這篇文章主要介紹了Jackson常用方法以及jacksonUtil工具類詳解,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2021-06-06
  • Java集合Set的簡單使用解析

    Java集合Set的簡單使用解析

    這篇文章主要介紹了Java集合Set的簡單使用解析,Set接口是Collection的子接口,Set接口相較于Collection接口沒有提供額外的方法,Set 集合不允許包含相同的元素,如果試把兩個相同的元素加入同一個 Set 集合中,則添加操作失敗,需要的朋友可以參考下
    2023-11-11
  • SpringBoot中ApplicationEvent和ApplicationListener用法小結(jié)

    SpringBoot中ApplicationEvent和ApplicationListener用法小結(jié)

    這篇文章介紹SpringBoot中ApplicationEvent用法,注意ApplicationEvent和MQ隊列雖然實現(xiàn)的功能相似,但是MQ還是有其不可替代性的,最本質(zhì)的區(qū)別就是MQ可以用于不同系統(tǒng)之間的消息發(fā)布,而SpringEvent這種模式只能在一個系統(tǒng)中,需要的朋友可以參考下
    2023-03-03
  • Mybatis Integer類型參數(shù)值為0時得到為空的解決方法

    Mybatis Integer類型參數(shù)值為0時得到為空的解決方法

    這篇文章主要介紹了Mybatis Integer類型參數(shù)值為0時得到為空的解決方法,有需要的朋友們可以學(xué)習(xí)下。
    2019-08-08
  • JAVA構(gòu)造器是否為靜態(tài)方法你知道嗎

    JAVA構(gòu)造器是否為靜態(tài)方法你知道嗎

    這篇文章主要為大家詳細(xì)介紹了JAVA構(gòu)造器是否為靜態(tài)方法,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來幫助
    2022-02-02
  • 深入理解Java 線程通信

    深入理解Java 線程通信

    這篇文章主要介紹了Java 線程通信的的相關(guān)資料,文中講解非常細(xì)致,代碼幫助大家更好的理解和學(xué)習(xí),感興趣的朋友可以了解下
    2020-06-06
  • springboot響應(yīng)json?null值過濾方式

    springboot響應(yīng)json?null值過濾方式

    這篇文章主要介紹了springboot響應(yīng)json?null值過濾方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2021-11-11

最新評論