对于spring之IOC种核心成员Bean,你有多少了解呢?

Spring 之 Bean

本节将学习一下 IOC 中的核心成员 Bean。

Bean 是一个个JAVA 类的实例(在Spring IOC 容器中的体现)。学习Bean首先我们从 IOC Container 和 Bean 之间的关系开始。

IOC Container & Bean

下面我们将通过一张图来展示 IOC Container 与 Bean的关系

由上图我们可知:

IOC Container 需要被初始化创建

IOC Container 主要有两种创建方式即本地创建个委托Web容器创建。

本地创建

所谓的本地创建就是直接通过代码创建,如下所示

ApplicationContext context =newClassPathXmlApplicationContext("ApplicationContext.xml");

Customercustomer = (Customer) context.getBean("customer");

委托Web 容器创建

所谓的委托Web 容器创建就是在WEB的应用程序中,WEB Servlet 容器创建好之后“自动”读取相应的 web.xml(泛指大类的web.xml 而非特指Tomcat中的web.xml),这个“自动”的动作实际有ConextLoaderListener 监听着,同时要在web.xml 中配置如下代码

<context-param>

contextConfigLocation

classpath:ApplicationContext.xml

</context-param>

<listener>

org.springframework.web.context.ContextLoaderListener

</listener>

以上两中方式都指明了Bean 配置的文件以及文件位置,一般我们默认为 classpath下。

Bean 的读取和Bean实例的创建需在 IOC Container 被创建在之后且创建的Bean实例只在 IOC Container 内。

在 IOC Container 被初始化创建之后会读取指定的Bean 配置文件,然后创建Bean 实例并将其注入至IOC Container 内以供程序调用。

Bean 的注入属性值

在Bean 属性的注入时我们也需要注意一些细节:

Bean Value值的注入

字面值:可用字符串表示的值,可以通过 元素标签或 value 属性进行注入。

基本数据类型及其封装类、String 等类型都可以采取字面值注入的方式

若字面值中包含特殊字符,可以使用《 ![CDATA[]]> 把字面值包裹起来

一个栗子

]]>

</property>

</bean>

Bean的引用

组成应用程序的 Bean 经常需要相互协作以完成应用程序的功能. 要使 Bean 能够相互访问, 就必须在 Bean 配置文件中指定对 Bean 的引用。

在 Bean 的配置文件中, 可以通过 元素或 ref 属性为 Bean 的属性或构造器参数指定对 Bean 的引用。

一个栗子

</util:list>

</bean>

Bean 的内部Bean

在一个Bean A的内部我们可以定义一个私有的Bean B。此时B 将是A私有的不能被其他Bean 引用或继承。同时B可以不用id 做唯一标识。

当 Bean 实例仅仅给一个特定的属性使用时, 可以将其声明为内部 Bean. 内部 Bean 声明直接包含在 或 元素里, 不需要设置任何 id 或 name 属性

内部 Bean 不能使用在任何其他地方

一个栗子

</bean>

</property>

</bean>

Bean 的 utility scheme

使用基本的集合标签定义集合时, 不能将集合作为独立的 Bean 定义, 导致其他 Bean 无法引用该集合, 所以无法在不同 Bean 之间共享集合。

可以使用 util schema 里的集合标签定义独立的集合 Bean. 需要注意的是, 必须在 根元素里添加 util schema 定义。

一个栗子

</util:list>

</bean>

Bean 的p 命名空间

为了简化 XML 文件的配置,越来越多的 XML 文件采用属性而非子元素配置信息。

Spring 从 2.5 版本开始引入了一个新的 p 命名空间,可以通过 元素属性的方式配置 Bean 的属性。

使用 p 命名空间后,基于 XML 的配置方式将进一步简化。

p 命名空间支持ref 引用和直接注入两种赋值方式

一个栗子

</bean>

Bean 的自动装配

我们可以在xml中配置bean 的自动装配。

Spring IOC 容器可以自动装配 Bean. 需要做的仅仅是在 的 autowire 属性里指定自动装配的模式

byType(根据类型自动装配): 若 IOC 容器中有多个与目标 Bean 类型一致的 Bean. 在这种情况下, Spring 将无法判定哪个 Bean 最合适该属性, 所以不能执行自动装配

byName(根据名称自动装配): 必须将目标 Bean 的名称和属性名设置的完全相同.

constructor(通过构造器自动装配): 当 Bean 中存在多个构造器时, 此种自动装配方式将会很复杂. 不推荐使用,缺点如下:

在 Bean 配置文件里设置 autowire 属性进行自动装配将会装配 Bean 的所有属性. 然而, 若只希望装配个别属性时, autowire 属性就不够灵活了

autowire 属性要么根据类型自动装配, 要么根据名称自动装配, 不能两者兼而有之

一般情况下,在实际的项目中很少使用自动装配功能,因为和自动装配功能所带来的好处比起来,明确清晰的配置文档更有说服力一些。

Bean 的继承

Spring 允许继承 bean 的配置, 被继承的 bean 称为父 bean. 继承这个父 Bean 的 Bean 称为子 Bean

子 Bean 从父 Bean 中继承配置, 包括 Bean 的属性配置

子 Bean 也可以覆盖从父 Bean 继承过来的配置

父 Bean 可以作为配置模板, 也可以作为 Bean 实例. 若只想把父 Bean 作为模板, 可以设置 的abstract 属性为 true, 这样 Spring 将不会实例化这个

并不是 元素里的所有属性都会被继承. 比如: autowire, abstract 等

也可以忽略父 Bean 的 class 属性, 让子 Bean 指定自己的类, 而共享相同的属性配置. 但此时 abstract 必须设为 true

一个栗子

Bean 的依赖

Spring 允许用户通过 depends-on 属性设定 Bean 前置依赖的Bean,前置依赖的 Bean 会在本 Bean 实例化之前创建好

如果前置依赖于多个 Bean,则可以通过逗号,空格或的方式配置 Bean 的名称

Abstract=true 的bean 不能被依赖

一个栗子

Bean 的作用域

在 Spring 中, 可以在 元素的 scope 属性里设置 Bean 的作用域。

默认情况下, Spring 只为每个在 IOC 容器里声明的 Bean 创建唯一一个实例, 整个 IOC 容器范围内都能共享该实例:所有后续的 getBean() 调用和 Bean 引用都将返回这个唯一的 Bean 实例.该作用域被称为 singleton, 它是所有 Bean 的默认作用域。

具体说明

一个栗子

</bean>

Bean 引用外部属性文件

在配置文件里配置 Bean 时, 有时需要在 Bean 的配置里混入系统部署的细节信息(例如: 文件路径, 数据源配置信息等). 而这些部署细节实际上需要和 Bean 配置相分离。

Spring 提供了一个 PropertyPlaceholderConfigurer 的 BeanFactory 后置处理器, 这个处理器允许用户将 Bean 配置的部分内容外移到属性文件中. 可以在 Bean 配置文件里使用形式为 ${var} 的变量, PropertyPlaceholderConfigurer 从属性文件里加载属性, 并使用这些属性来替换变量。

Spring 还允许在属性文件中使用 ${propName},以实现属性之间的相互引用。

一个栗子

xml 文件

<!-- 配置属性文件 -->

class="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer">

<list>

/WEB-INF/classes/dbconfig.properties

/WEB-INF/classes/redis.properties

</list>

</property>

</bean>

<!-- 阿里 druid数据库连接池 -->

destroy-method="close">

<!-- 数据库基本信息配置 -->

<!-- 最大并发连接数 -->

<!-- 初始化连接数量 -->

<!-- 配置获取连接等待超时的时间 -->

<!-- 最小空闲连接数 -->

<!-- 配置间隔多久才进行一次检测,检测需要关闭的空闲连接,单位是毫秒 -->

<!-- 配置一个连接在池中最小生存的时间,单位是毫秒 -->

<!-- 打开removeAbandoned功能 -->

<!-- 1800秒,也就是30分钟 -->

<!-- 关闭abanded连接时输出错误日志 -->

</bean>

properties文件

url:jdbc:mysql://localhost:3306/numysql?useUnicode=true&characterEncoding=utf8&characterSetResults=utf8

driverClassName:com.mysql.jdbc.Driver

username:root

password:bai5331359

filters:stat

maxActive:20

initialSize:1

maxWait:60000

minIdle:10

maxIdle:15

timeBetweenEvictionRunsMillis:60000

minEvictableIdleTimeMillis:300000

validationQuery:SELECT'x'

testWhileIdle:true

testOnBorrow:false

testOnReturn:false

maxOpenPreparedStatements:20

removeAbandoned:true

removeAbandonedTimeout:1800

logAbandoned:true

Bean 的SpEL

SpEL:Spring表达式语言。

Spring 表达式语言(简称SpEL):是一个支持运行时查询和操作对象图的强大的表达式语言。

语法类似于 EL:SpEL 使用 #{…} 作为定界符,所有在大框号中的字符都将被认为是 SpEL。

SpEL 为 bean 的属性进行动态赋值提供了便利。

通过 SpEL 可以实现

 通过 bean 的 id 对 bean 进行引用

 调用方法以及引用对象中的属性

 计算表达式的值

 正则表达式的匹配

SpEL字面量的表示

整数:property name=”count” value=”#{5}”/>

小数:property name=”frequency” value=”#{89.7}”/>

科学计数法:property name=”capacity” value=”#{1e4}”/>

String可以使用单引号或者双引号作为字符串的定界符号:property name=“name” value=”#{‘Chuck’}”/> 或 property name=’name’ value=’#{“Chuck”}’/>

Boolean:property name=”enabled” value=”#{false}”/>

SpEL:引用 Bean、属性和方法

引用其他对象

</util:list>

引用其他对象的属性

</bean>

</bean>

调用其他方法,还可以链式操作

</bean>

以上栗子为 Bean orderFour 调用 Bean orderFive 的 返回值并将该返回值转为大写。

SpEL支持的运算符号

算数运算符:+, -, *, /, %, ^:

</bean>

</bean>

加号还可以用作字符串连接

</bean>

</bean>

比较运算符: <, >, ==, <=, >=, lt, gt, eq, le, ge

逻辑运算符号: and, or, not, |

if-else 运算符:?: (ternary), ?: (Elvis)

if-else 的变体:三部运算法

正则表达式:matches

SpEL:引用 Bean、属性和方法

Bean 的生命周期

IOC 容器中 Bean 的生命周期方法。

Spring IOC 容器可以管理 Bean 的生命周期, Spring 允许在 Bean 生命周期的特定点执行定制的任务。

Spring IOC 容器对 Bean 的生命周期进行管理的过程

通过构造器或工厂方法创建 Bean 实例

为 Bean 的属性设置值和对其他 Bean 的引用

调用 Bean 的初始化方法

Bean 可以使用了

当容器关闭时, 调用 Bean 的销毁方法

在 Bean 的声明里设置 init-method 和 destroy-method 属性, 为 Bean 指定初始化和销毁方法。

一个栗子

java 文件部分示例

publicvoidinitExample(){

System.out.println("order Bean create initFunction");

    }

publicvoiddestroyExampl(){

System.out.println("order Bean destroyed function");

    }

Bean

</bean>

结果

Bean 后置处理器

Bean 后置处理器允许在调用初始化方法前后对 Bean 进行额外的处理。

Bean 后置处理器对 IOC 容器里的所有 Bean 实例逐一处理, 而非单一实例. 其典型应用是: 检查 Bean 属性的正确性或根据特定的标准更改 Bean 的属性。

对Bean 后置处理器而言, 需要实现 BeanPostProcessor 接口. 在初始化方法被调用前后, Spring 将把每个 Bean 实例分别传递给上述接口的以下两个方法。

postProcessAfterInitialization

postProcessBeforeInitialization

添加 Bean 后置处理器后 Bean 的生命周期

Spring IOC 容器对 Bean 的生命周期进行管理的过程

通过构造器或工厂方法创建 Bean 实例

为 Bean 的属性设置值和对其他 Bean 的引用

将 Bean 实例传递给 Bean 后置处理器的 

postProcessBeforeInitialization 方法

调用 Bean 的初始化方法

将 Bean 实例传递给 Bean 后置处理器的 

postProcessAfterInitialization方法

Bean 可以使用了

当容器关闭时, 调用 Bean 的销毁方法

一个栗子

java 部分代码

packagecom.spring.example.utils;

importorg.springframework.beans.BeansException;

importorg.springframework.beans.factory.config.BeanPostProcessor;

importcom.spring.example.entiry.Orders;

publicclassMyBeanPostProcessorimplementsBeanPostProcessor{

//该方法在 init 方法之后被调用

    @Override

publicObjectpostProcessAfterInitialization(Objectarg0,Stringarg1)

throwsBeansException {

if(arg1.equals("orderThree")){

System.out.println("postProcessAfterInitialization..."+ arg0 +","+ arg1);

Ordersorder = (Orders) arg0;

order.setOrderId("txAdd4541548dd12");;

        }

returnarg0;

    }

//该方法在 init 方法之前被调用

//可以工作返回的对象来决定最终返回给 getBean 方法的对象是哪一个, 属性值是什么

        /**

*@paramarg0: 实际要返回的对象

*@paramarg1: bean 的 id 值

        */

    @Override

publicObjectpostProcessBeforeInitialization(Objectarg0,Stringarg1)

throwsBeansException {

if(arg1.equals("orderThree"))

System.out.println("postProcessBeforeInitialization..."+ arg0 +","+ arg1);

returnarg0;

    }

}

ProcessBean 配置

<!-- 配置 bean 后置处理器: 不需要配置 id 属性, IOC 容器会识别到他是一个 bean 后置处理器, 并调用其方法 -->

结果示例

列表内容

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

推荐阅读更多精彩内容