Spring高级配置之运行时注入

之前,我们设置Bean的属性值时,采用的都是硬编码的形式。比如,在定义BlankDisc时:

@Bean
public BlankDisc blankDisc(){
    
    return new BlankDisc("This is BlankDisc", "The Beatles");
}
 

与之类似,采用XML的方式也是硬编码:

<bean id = "blankDisc" class = "com.inject.service.BlankDisc"
c:_title = "This id BlankDisc"
c:_artist = "This Beatles" />     

硬编码的坏处想必大家都知道,所以我们就想能不能让这些值在程序运行时再确定。其实Spring已经提供了两种方式实现运行时注入的功能:

  • 属性占位符(Property planceholder)
  • Spring 表达式语言(SpEL)

1.使用属性占位符注入外部的值

@Autowired
Environment env;

public BlankDisc blanKDiscBean2(){
    return new BlankDisc(
            env.getProperty("disc.title"),
            env.getProperty("disc.artist"));
}

1.1深入学习Spring的Environment

Environment的getProperty()方法有四个重载方法:

  • String getProperty(String key)
  • String getProperty(String key,String defaultValue)
  • T getProperty(String key, Class<T> type)
  • T getProperty(String key, Class<T> type,T defaultValue)

前两种形式的getProperty()方法都是返回String类型的值。后两种重载方法不是直接返回字符串,而是将获取到的值转化为指定类型的值,比如获取连接池中维持连接的总数量:

int connectionCount = env.getProperty("db.connoctionCount",Integer.class,30);

在使用getProperty()获取属性时,如果这个属性没有定义,那他获取的是null,如果我们不想让这样的情况发生,可以使用getRequiredProperty()方法:

//必须定义参数
env.getRequiredProperty("disc.title"),
env.getRequiredProperty("disc.artist"));

如果disc.title或disc.arist没有定义的话,将会抛出IllegalStateException异常。
如果你想检查某个属性是否存在,那么可以调用containsProperty()方法:

boolean titleExists = env.containsProperty(“disc.title”);

如果你想将属性解析为类的话,可以使用getPropertyAsClass()方法:

Class<BlankDisc> blankDisc = env.getPropertyAsClass("disc.class",BlankDisc.class);

除了属性相关的功能外,Environment还提供了一些方法来检查哪些profile处于激活状态:

  • String [] getActiveProfiles():返回激活profile名称的数组;
  • String [] getDefaultProfile():返回默认profile名称的数组;
  • boolean acceptsPrifiles(String ... profiles):如果environment支持给定的profile返回true。

1.2解析属性占位符

在Spring装配中,占位符的形式为“${...}”包装的属性名称。下面我们使用属性占位符在XML中解析tomcat连接池bean的属性:

    <!-- 数据源配置, 使用Tomcat JDBC连接池 -->
    <bean id="dataSource" class="org.apache.tomcat.jdbc.pool.DataSource" destroy-method="close">
        <!-- Connection Info -->
        <property name="driverClassName" value="${jdbc.driver}"/>
        <property name="url" value="${jdbc.url}"/>
        <property name="username" value="${jdbc.username}"/>
        <property name="password" value="${jdbc.password}"/>

        <!-- Connection Pooling Info -->
        <property name="maxActive" value="${jdbc.pool.maxActive}"/>
        <property name="maxIdle" value="${jdbc.pool.maxIdle}"/>
    </bean>

要使用占位符,需要配置PropertySourcesPlaceholderConfigurer bean:

@Bean
public static PropertySourcesPlaceholderConfigurer placehoderConfigurer(){
    
    return new PropertySourcesPlaceholderConfigurer();
}

在Xml配置要使用到命名空间中的<context:property-placeholder>元素,它可以为我们生成PropertySourcesPlaceholderConfigurer bean:

<context:property-placeholder ignore-unresolvable="true"
                   location="classpath*:/application.properties,
                   classpath*:/application.local.properties" file-encoding="UTF-8"/>

2.使用Spring表达式语言进行配置

Spring 3引入了Spring表达式(Spring Excpression Language,SpEL),他能够以一种强大和简洁的方式将值装配到bean属性和构造器参数中,这个过程中所使用的表达式会在运行时计算得到值。SpEL的特性有以下几点:

  • 使用bean的ID来引用bean
  • 调用方法和访问对象的属性
  • 对值进行算术、关系和逻辑运算
  • 正则表达式匹配
  • 集合操作

2.1SpEl表达式样例

格式 说明
#{1} 得到的值就是1
#{T(System).currentTimeMillis() 获取当前时间的毫秒值,T()表达式会将java.lang.System视为java中对应的类型,通过.方法调用对应的静态方法
#{blankDisc.title} 获取ID为blankDisc bean的属性值
#{systemPropertys['disc.title']} 通过systemPropertys对象引用系统属性

上面几个只是几个基本的SpEl样例,后面我们会着重介绍,下面我们先看看使用SpEL表达式装配bean的属性:

public BlankDisc(
            @Value("#{systemPropertys['disc.title']}") String title, 
            @Value("#{systemPropertys['disc.artist']}") String artist){
        this.title = title;
        this.artist = artist;
    }

上例中使用到@Value注解,在注解中设置SpEL表达式。上面我们学习了几个简单的样例,也学习了如何将SpEL表达式解析得到的值注入到bean中,那么我们来继续学习一下SpEl表达式支持的基础表达式吧。

2.2表示字面值

使用SpEL字面量样式可以表示整数、浮点数、String以及Boolean类型的值:

#{1}
#{3.14159}
#{9.87E4}
#{'Hello'}
#{false}

2.3引入bean属性和方法

SpEL能够通过ID引入其他的bean、bean的属性和bean的方法:

#{blankDisc}
#{blankDisc.title}
#{blankDisc.sayHello()}

如果还可以对方法的返回值做处理,直接调用返回值类型的属性或方法,假设sayHello()返回值是String类型:#{blankDisc.sayHello().toUpperCase()},这样就可以获取返回值的大写字母形式。
但如果返回值是null,调用它的方法就会报空指针异常,为了预防这种情况,可以使用 ? 判断获得返回值是否为null:#{blankDisc.sayHello()?.toUpperCase()}如果sayHello()方法返回值是null,SpEl将不会调用toUpperCase()方法,SpEL的返回值是null。

2.4在表达式中使用类型

如果SpEL表达式访问类作用域的方法或常量时,要使用到T()这个关键的运算符:#{T(java.lang.Math)}运算符的结果是一个Class对象,它代表了java.lang.Math。通过它我们可以访问Math的静态方法和常量。

#{T(java.lang.Math).PI}
#{T(java.lang.Math).random()}

2.5SpEL运算符

SpEL提供了多个运算符,这些运算符可以使用到SpEL表达式上:

运算符类型 运算符
算术运算符 +、- 、*、/ 、%、^
比较运算符 <、>、 = 、<=、 >=、 lt、 gt、 eq、 ge
条件运算符 ?:(ternary)、 ?:(Elvis)
逻辑运算符 and、 or、not、
正则表达式 matches

2.6计算正则表达式

SpEL通过matches运算符支持表达式中的模式匹配,匹配成功返回true,否则否则返回false,下面我们判断邮箱是否符合正则表达式:

#{admin.email matches '[a-zA-Z0-9.-%+-]+@[a-zA-Z0-9.-]+\\.com'}

2.7计算集合

使用SpEL引用列表中的一个元素:#{jubox.songs[4].title},获取列表的第5个元素(基于0开始),[] 是从集合或数组中按照索引取元素。
SpEL提供了查询运算符(.?[]),他会用来对集合进行过滤,得到集合的一个子集。下面我们将Aerosmith的歌曲过滤出来:

#{jukebox.songs.?[artist eq 'Aerosmith']}

.^[]查询第一个匹配项
.$[]查询最后一个匹配项

投影运算符(.![]),将集合对象特定的字段投影到另一个集合中,下面我们将歌曲的名称投影到另一个String类型的集合中:

#{jukebox.songs.![title]}

我们还可以过滤要投影的歌曲,下面我们获取Aerosmith歌曲的title:

#{jukebox.songs.?[artist eq 'Aerosmith'.![title]]}

自此运行时注入的全部内容已经全部介绍完啦。终于完了啦,下个月我们开始学习Spring的Aop机制,很期待吧,come on!

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,287评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,346评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,277评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,132评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,147评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,106评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,019评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,862评论 0 274
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,301评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,521评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,682评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,405评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,996评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,651评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,803评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,674评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,563评论 2 352

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,651评论 18 139
  • Spring Boot 参考指南 介绍 转载自:https://www.gitbook.com/book/qbgb...
    毛宇鹏阅读 46,803评论 6 342
  • 本章内容: Spring profile 条件化的bean声明 自动装配与歧义性 bean的作用域 Spring表...
    谢随安阅读 1,187评论 0 5
  • “近朱者赤,近墨者黑”,“孟母三迁”,“蓬生麻中,不扶自直”,这些成语都像我们阐释了一个道理——朋友圈对一个人的成...
    英语老师Audrey阅读 1,547评论 3 43
  • 今天跟老公聊天聊到了一个现象。这次我和孩子们回老家十多天,刚开始老公还以为他会有很多精彩的安排,结果后来也发现自己...
    敏敏52阅读 380评论 0 0