电脑知识|欧美黑人一区二区三区|软件|欧美黑人一级爽快片淫片高清|系统|欧美黑人狂野猛交老妇|数据库|服务器|编程开发|网络运营|知识问答|技术教程文章 - 好吧啦网

您的位置:首頁技術文章
文章詳情頁

SpringBoot SpEL語法掃盲與查詢手冊的實現

瀏覽:4日期:2023-05-16 14:46:27

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

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

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

I. 語法百科

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

1. 字面表達式

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 2147483647int maxValue = (Integer) parser.parseExpression('0x7FFFFFFF').getValue();boolean trueValue = (Boolean) parser.parseExpression('true').getValue();Object nullValue = parser.parseExpression('null').getValue();

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

輸出結果

str: Hello Worlddouble: 6.0221415E23int: 2147483647bool: truenull: null

2. Inline List

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

ExpressionParser parser = new SpelExpressionParser();// Integer列表List numbers = (List) parser.parseExpression('{1,2,3,4}').getValue();System.out.println('list: ' + numbers);// List的元素為ListList<List> listlOfLists = (List) parser.parseExpression('{{’a’,’b’},{’x’,’y’}}').getValue();System.out.println('List<List> : ' + listlOfLists);

輸出結果

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

3. Inline map

{key:value}來表示 map 表達式,空 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);}

輸出結果

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

4. 數組

數組可以借助new構造方法來實現,通過下標ary[index]的方式訪問數組中的元素

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); // 通過下標訪問數組中的元素 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. 表達式

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

下面給出一些簡單的實例

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進行包裝 System.out.println('instance : ' + parser.parseExpression('’a’ instanceof T(String)').getValue()); //正則表達式 System.out.println('22 是否為兩位數字 :' + parser.parseExpression('22 matches ’d{2}’').getValue());}

輸出結果

1+2= 31<2= truetrue ? hello : false > helloinstance : true22 是否為兩位數字 :true

6. Type 與靜態類

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

比如我們有一個靜態類

public static class StaClz { public static String txt = '靜態屬性'; public static String hello(String tag) { return txt + ' : ' + tag; }}

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

public void type() { // class,靜態類 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());}

輸出結果如下

txt: 靜態屬性static method return: 靜態屬性 : 一灰灰blogclass: java.lang.String

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

7. 構造方法

上面介紹 array 的時候,就介紹了使用new來創建數組對象,當然也可以直接構造其他的普通對象, 如我們新建一個測試類

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 創建一個對象的實例

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);}

輸出結果如下:

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

請注意,構造方法中類的完整簽名

8. 變量引用

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

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

一般的操作過程如下:

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

一個簡單的實例

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);}

輸出結果如下

variable name: 一灰灰blogvariable age: 18

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

9. 函數

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

public void function() { try { ExpressionParser parser = new SpelExpressionParser(); EvaluationContext context = SimpleEvaluationContext.forReadOnlyDataBinding().build(); // 注冊一個方法變量,參數為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(); }}

輸出結果如下

function call: 靜態屬性 : 一灰灰

10. bean 訪問

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

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

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

@Data@Componentpublic 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;@Overridepublic 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);}

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

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 支持三元表達式,在上述的表達式中也給出了實例

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

輸出結果如下

ifTheElse: 正確

12. elvis

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

這個也屬于我們經常遇到的一種場景,如果 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);}

輸出結果如下

elvis-before Unknownelvis-after Exists!

13. 安全表達式

在 java 中,最常見最討厭的是一個就是 NPE 的問題,SpEL 中當然也可能出現這種情況,但是若在 SpEL 中進行非空判斷,那就很不優雅了,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);}

輸出結果如下

safeOperate-before: nullsafeOperate-after: 7

14. 容器截取

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

xx.?[expression], 請注意中括弧中的表達式必須返回 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); // 表達式內部用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);}

輸出結果如下

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

注意

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

15. 容器映射

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

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

舉例如下

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);}

輸出結果如下:

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

16. 表達式模板

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

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

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

舉例說明

public void template() { // 模板,混合字面文本與表達式,使用 #{} 將表達式包裹起來 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);}

輸出結果如下

template: random number is 0.10438946298113871

17. 小結

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

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

II. 其他

0. 項目

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

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

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持好吧啦網。

標簽: Spring
相關文章:
主站蜘蛛池模板: 包塑丝_高铁绑丝_地暖绑丝_涂塑丝_塑料皮铁丝_河北创筹金属丝网制品有限公司 | 据信,上课带着跳 D 体验-别样的课堂刺激感受引发网友热议 | PCB厂|线路板厂|深圳线路板厂|软硬结合板厂|电路板生产厂家|线路板|深圳电路板厂家|铝基板厂家|深联电路-专业生产PCB研发制造 | 电磁铁_小型推拉电磁铁_电磁阀厂家-深圳市宗泰电机有限公司 | 天津散热器_天津暖气片_天津安尼威尔散热器制造有限公司 | 湖南档案密集架,智能,物证,移动,价格-湖南档案密集架厂家 | 高扬程排污泵_隔膜泵_磁力泵_节能自吸离心水泵厂家-【上海博洋】 | [品牌官网]贵州遵义双宁口腔连锁_贵州遵义牙科医院哪家好_种植牙_牙齿矫正_原华美口腔 | 吹田功率计-长创耐压测试仪-深圳市新朗普电子科技有限公司 | 天津货架厂_穿梭车货架_重型仓储货架_阁楼货架定制-天津钢力仓储货架生产厂家_天津钢力智能仓储装备 | 重庆钣金加工厂家首页-专业定做监控电视墙_操作台 | 滤芯,过滤器,滤油机,贺德克滤芯,精密滤芯_新乡市宇清流体净化技术有限公司 | 盘式曝气器-微孔曝气器-管式曝气器-曝气盘-斜管填料 | 郑州市前程水处理有限公司 | 杰恒蠕动泵-蠕动泵专业厂家-19年专注蠕动泵 | 冷水机,风冷冷水机,水冷冷水机,螺杆冷水机专业制造商-上海祝松机械有限公司 | 焊接减速机箱体,减速机箱体加工-淄博博山泽坤机械厂 | 黑田精工电磁阀-CAMMOZI气缸-ROSS电磁-上海茂硕机械设备有限公司 | 挤塑板-XPS挤塑板-挤塑板设备厂家[襄阳欧格] | 江苏密集柜_电动_手动_移动_盛隆柜业江苏档案密集柜厂家 | 扒渣机厂家_扒渣机价格_矿用扒渣机_铣挖机_撬毛台车_襄阳永力通扒渣机公司 | 气弹簧定制-气动杆-可控气弹簧-不锈钢阻尼器-工业气弹簧-可调节气弹簧厂家-常州巨腾气弹簧供应商 | 钢制拖链生产厂家-全封闭钢制拖链-能源钢铝拖链-工程塑料拖链-河北汉洋机械制造有限公司 | 地源热泵一体机,地源热泵厂家-淄博汇能环保设备有限公司 | 航空连接器,航空插头,航空插座,航空接插件,航插_深圳鸿万科 | sfp光模块,高速万兆光模块工厂-性价比更高的光纤模块制造商-武汉恒泰通 | 恒温槽_恒温水槽_恒温水浴槽-上海方瑞仪器有限公司 | 福建珂朗雅装饰材料有限公司「官方网站」 | 丁基胶边来料加工,医用活塞边角料加工,异戊二烯橡胶边来料加工-河北盛唐橡胶制品有限公司 | 耐磨陶瓷,耐磨陶瓷管道_厂家-淄博拓创陶瓷科技 | 广东教师资格网-广东教师资格证考试网 | 氢氧化钙设备_厂家-淄博工贸有限公司 | 产业规划_产业园区规划-产业投资选址及规划招商托管一体化服务商-中机院产业园区规划网 | 中高频感应加热设备|高频淬火设备|超音频感应加热电源|不锈钢管光亮退火机|真空管烤消设备 - 郑州蓝硕工业炉设备有限公司 | 酵素生产厂家_酵素OEM_酵素加盟_酵素ODM_酵素原料厂家_厦门益力康 | 分子精馏/精馏设备生产厂家-分子蒸馏工艺实验-新诺舜尧(天津)化工设备有限公司 | 上海公司注册-代理记账-招投标审计-上海昆仑扇财税咨询有限公司 上海冠顶工业设备有限公司-隧道炉,烘箱,UV固化机,涂装设备,高温炉,工业机器人生产厂家 | 马尔表面粗糙度仪-MAHR-T500Hommel-Mitutoyo粗糙度仪-笃挚仪器 | 小小作文网_中小学优秀作文范文大全 | 江苏农村商业银行招聘网_2024江苏农商行考试指南_江苏农商行校园招聘 | 黄石妇科医院_黄石东方女子医院_黄石东方妇产医院怎么样 | 断桥铝破碎机_铝合金破碎机_废铁金属破碎机-河南鑫世昌机械制造有限公司 |