什麼是SpringEL(SpEL)?
Spring3中引入了Spring表達式語言―SpringEL,SpEL是一種強大,簡潔的組裝Bean的方式
SpringEL可以透過運行期間執行的表達式將值組裝到我們的屬性或建構子當中
#SpringEL可以呼叫JDK中提供的靜態常數,取得外部Properties檔案中的的配置
為什麼要使用SpringEL?
平常透過設定檔或Annotaton注入的Bean,其實都可以稱為靜態性注入
如Bean A中有變數A,它的值需要依照Bean B的B變數為參考,在這場景下靜態注入就對這樣的處理顯得非常無力
而Spring3增加的SpringEL就可以完全滿足這種需求,而且還可以對不同Bean的字段進行計算再進行賦值,功能非常強大
如何使用SpringEL?
SpringEL從名字來看就能看出和EL是有點關係的,SpringEL的使用和EL表達式的使用非常相似
EL表達式在JSP頁面更方便的取得後台中的值,而SpringEL就是為了更方便取得Spring容器中的Bean的值
EL使用${},而SpringEL使用#{}進行表達式的宣告
兩者主要差異
#$是去找外部設定的參數,將值賦過來
#是SpEL表達式,去尋找對應變數的內容
也可以直接使用@value("常數")注入不使用EL ,這樣寫法與直接賦值等價
如果是在Spring中使用可以使用**@PropertySource("classpath:my.properties")**載入對應設定檔
# 配置文件 com: codecoord: el: num: 1001 name: el language: - java - spring - mysql - linux # 逗号分隔可以注入列表 language02: java,spring,mysql,linux
使用EL注入簡單值
/** * 注入简单值,直接注入不使用EL,EL不支持直接指定常量 * 直接在EL中指定的常量会当做配置处理,和直接赋值等价 */ @Value("1432516744") private Integer no;
注入設定檔屬性值
/** * 注入整型属性值 */ @Value("${com.codecoord.el.num}") private Integer num; /** * 注入字符属性值 */ @Value("${com.codecoord.el.name}") private String name;
注入預設值
/** * 注入字符不存在属性值并指定默认值,默认值使用过冒号分隔 : * 注入常量其实就可以指定一个不存在的配置然后使用默认值,此处skill的值为java */ @Value("${com.codecoord.el.skill:java}") private String skill;
注入清單
不支援直接設定檔中陣列語法格式註入清單
可以辨識使用逗號,分隔的配置,spring預設以,分隔
// 错误写法:不支持直接注入yml列表格式语法列表 @Value("${com.codecoord.el.language}") private List<String> listLanguage; @Value("${com.codecoord.el.language}") private String[] strLanguage;
/** * 支持,分隔的注入列表 */ @Value("${com.codecoord.el.language02}") private List<String> listLanguage02; @Value("${com.codecoord.el.language02}") private String[] strLanguage02;
完整參考如下
設定檔
server: port: 8888 com: codecoord: el: num: 1001 name: el language: - java - spring - mysql - linux # 逗号分隔可以注入列表 language02: java,spring,mysql,linux
屬性配置類別
import lombok.Data; import org.springframework.beans.factory.annotation.Value; import org.springframework.stereotype.Component; import java.util.List; @Data @Component public class ElConfig { /** * 注入简单值,直接注入不使用EL,EL不支持直接指定常量 * 直接在EL中指定的常量会当做配置处理,和直接赋值等价 */ @Value("1432516744") private Integer no; /** * 注入整型属性值 */ @Value("${com.codecoord.el.num}") private Integer num; /** * 注入字符属性值 */ @Value("${com.codecoord.el.name}") private String name; /** * 注入字符不存在属性值并指定默认值,默认值使用过冒号分隔 : * 注入常量其实就可以指定一个不存在的配置然后使用默认值,此处skill的值为java */ @Value("${com.codecoord.el.skill:java}") private String skill; /// 不支持直接注入列表 /*@Value("${com.codecoord.el.language}") private List<String> listLanguage; @Value("${com.codecoord.el.language}") private String[] strLanguage;*/ /** * 支持,分隔的注入列表 */ @Value("${com.codecoord.el.language02}") private List<String> listLanguage02; @Value("${com.codecoord.el.language02}") private String[] strLanguage02; }
#向controller中注入配置類,然後存取介面測試結果如下
{ "no": 1432516744, "num": 1001, "name": "el", "skill": "java", "listLanguage02": [ "java", "spring", "mysql", "linux" ], "strLanguage02": [ "java", "spring", "mysql", "linux" ] }
1、使用SpEL注入簡單值和普通EL注入使用基本一致
2、SpEl注入map
設定檔中需要使用雙引號括起來,否則將會注入失敗,key為單引號
# SpEl spEl: mapInject: "{"name": "SpEl", "website": "http://www.codeocord.com"}"
java類別中先使用${spEl.mapInject}注入字串值,#{}會解析字串的值轉為map
@Value("#{${spEl.mapInject}}") private Map<String, String> mapInject;
3、SpEl注入list
除了可以透過EL注入listI外,也可以使用#{${}.split("分隔符號")}的方式註入List
在設定檔中例如使用#分隔
spEl: listInject: "44#11#99#100"
java類別中先使用${spEl.listInject}注入字串值,內容使用單引號括起來,然後對字串使用split方法分隔
提示:避免為空情況,可以給一個預設值空串
@Value("#{"${spEl.listInject:}".split("#")}") private List<String> listInject;
4、動態注入
上述注入都是靜態注入,SpEl支援從Spring容器中註入訊息,稱為動態注入。動態注入類別如下
import lombok.AllArgsConstructor; import lombok.Data; import org.springframework.stereotype.Component; import java.util.ArrayList; import java.util.HashMap; import java.util.List; import java.util.Map; @Component @Data public class SpElConstant { private String name = "SpElConstant-name"; private String nickname = "tianxin"; private int num = 100; private List<String> product = new ArrayList<String>() {{ add("huaweiMate30Pro"); add("xiaomi10x5g"); }}; private Map<String, String> productMap = new HashMap<String, String>() {{ put("huaweiMate30Pro", "5999"); put("xiaomi10x5g", "4999"); }}; private List<City> cityList = new ArrayList<City>() {{ add(new City("深圳", 1000L)); add(new City("杭州", 2000L)); add(new City("贵阳", 900L)); }}; public String showProperty() { return "showProperty-无参数"; } public String showProperty(String name) { return "showProperty-" + name; } @Data @AllArgsConstructor static class City { private String name; private long population; } }
SpEl支援與不支援操作
#支援動態注入實例,類似物件自動注入
#SPL不支援直接注入設定檔中的配置
支援呼叫靜態和實例方法
靜態方法:@Value( "#{T(package.ClassName).ConstFieldName")#支援呼叫靜態類別或常數
支援運算子運算
支援操作集合
支援查詢篩選集合和投影
#########注入完整操作如下# ##import lombok.Data; import org.springframework.beans.factory.annotation.Value; import org.springframework.stereotype.Component; import java.util.List; import java.util.Map; @Data @Component public class SpElConfig { /// 不支持直接注入配置文件值 /*@Value("#{com.codecoord.el.num}") private Integer num;*/ /** * 对象注入 */ @Value("#{spElConstant}") private SpElConstant spElConstant; /** * 注入ID为spElConstant Bean中的STR常量/变量 */ @Value("#{spElConstant.name}") private String name; /** * 调用无参方法 */ @Value("#{spElConstant.showProperty()}") private String method1; /** * 有参接收字符串的方法 */ @Value("#{spElConstant.showProperty("Hell SpringEL")}") private String method2; /** * 方法返回的String为大写 */ @Value("#{spElConstant.showProperty().toUpperCase()}") private String method3; /** * 若使用method3这种方式,若果showProperty返回为null * 将会抛出NullPointerException,可以使用以下方式避免 * 使用?.符号代表若然左边的值为null,将不执行右边方法 */ @Value("#{spElConstant.showProperty()?.toUpperCase()}") private String method4; /** * 注入math常量 */ @Value("#{T(java.lang.Math).PI}") private double pi; /** * 用random方法获取返回值 */ @Value("#{T(java.lang.Math).random()}") private double random; /** * 获取文件路径符号 */ @Value("#{T(java.io.File).separator}") private String separator; /** * 拼接字符串 */ @Value("#{spElConstant.nickname + " " + spElConstant.name}") private String concatString; /** * 对数字类型进行运算,spElConstant拥有num属性 */ @Value("#{3 * T(java.lang.Math).PI + spElConstant.num}") private double operation; /** * 进行逻辑运算 */ @Value("#{spElConstant.num > 100 and spElConstant.num <= 200}") private boolean logicOperation; /** * 进行或非逻辑操作 */ @Value("#{not (spElConstant.num == 100) or spElConstant.num <= 200}") private boolean logicOperation2; /** * 使用三元运算符 */ @Value("#{spElConstant.num > 100 ? spElConstant.num : spElConstant.num + 100}") private Integer logicOperation3; /** * 获取下标为0的元素 */ @Value("#{spElConstant.product[0]}") private String str; /** * 获取下标为0元素的大写形式 */ @Value("#{spElConstant.product[0]?.toUpperCase()}") private String upperStr; /** * 获取map中key为hello的value */ @Value("#{spElConstant.productMap["hello"]}") private String mapValue; /** * 根据product下标为0元素作为key获取testMap的value */ @Value("#{spElConstant.productMap[spElConstant.product[0]]}") private String mapStrByproduct; /** * 注入人口大于等于1000人口的城市 */ @Value("#{spElConstant.cityList.?[population >= 1000]}") private List<SpElConstant.City> cityList; /** * 注入人口等于900人口的城市 */ @Value("#{spElConstant.cityList.?[population == 900]}") private SpElConstant.City city; /** * 注入人口大于等于1000人口的城市,且只保留城市名称 */ @Value("#{spElConstant.cityList.?[population >= 1000].![name]}") private List<String> cityName; }###注入結果###
{ "spElConstant": { "name": "SpElConstant-name", "nickname": "tianxin", "num": 100, "product": [ "huaweiMate30Pro", "xiaomi10x5g" ], "productMap": { "xiaomi10x5g": "4999", "huaweiMate30Pro": "5999" }, "cityList": [ { "name": "深圳", "population": 1000 }, { "name": "杭州", "population": 2000 }, { "name": "贵阳", "population": 900 } ] }, "name": "SpElConstant-name", "method1": "showProperty-无参数", "method2": "showProperty-Hell SpringEL", "method3": "SHOWPROPERTY-无参数", "method4": "SHOWPROPERTY-无参数", "pi": 3.141592653589793, "random": 0.19997238292235787, "separator": "", "concatString": "tianxin SpElConstant-name", "operation": 109.42477796076938, "logicOperation": false, "logicOperation2": true, "logicOperation3": 200, "str": "huaweiMate30Pro", "upperStr": "HUAWEIMATE30PRO", "mapValue": null, "mapStrByproduct": "5999", "cityList": [ { "name": "深圳", "population": 1000 }, { "name": "杭州", "population": 2000 } ], "city": { "name": "贵阳", "population": 900 }, "cityName": [ "深圳", "杭州" ] }###Spring操作外部Properties檔案###
<!-- 首先通过applicaContext.xml中<util:properties>增加properties文件 --> <!-- 注意需要引入Spring的util schemea命名空间和注意id属性,id属性将在SpringEL中使用 --> <util:properties id="db" location="classpath:application.properties"/>
public class TestSpringEL { // 注意db为xml文件中声明的id @Value("#{db["jdbc.url"]}") private String propertiesValue; }###SpringEL在使用時僅僅是一個字串,不易於排錯與測試,也沒有IDE檢查我們的語法,當出現錯誤時較難檢測,複雜的表達式不建議透過SpringEL方式註入。在非必要情況下,不建議使用SpEl的複雜注入,清晰可讀的程式碼更為重要且有利於排查問題######四、屬性自動注入######上述都是透過指定字段進行注入,可以透過@ConfigurationProperties指定前綴進行自動注入###
org.springframework.boot.context.properties.ConfigurationProperties###配置類別###
user: id: ${random.uuid} name: autowire address: unknown website: www.codecoord.com age: ${random.int}###自動屬性注入類別###########透過prefix指定前端為user ,然後將會把user.後面的型別依照名稱進行注入############注意必須提供setter方法###
import lombok.Data; import org.springframework.boot.context.properties.ConfigurationProperties; import org.springframework.stereotype.Component; @Component @ConfigurationProperties(prefix = "user") @Data public class UserConfig { private String id; private String name; private String address; private String website; private Integer age; }
可以通过@EnableConfigurationProperties(value = UserConfig.class)将UserConfig再次强制注入,问题出现在如果UserConfig为第三方jar包内的配置类,则可能出现属性没有注入情况,所以可以指定注入
以上是SpringBoot SpringEL表達式怎麼使用的詳細內容。更多資訊請關注PHP中文網其他相關文章!