这篇文章是对前一篇的一些补充:

1.SpringIOC容器可以管理Bean的生命周期:

  • 通过构造器或工厂方法创建bean的实例;
  • 为bean属性设置值或者引入其他bean;
  • 调用bean的初始化方法,此时bean就可以使用了;
  • 容器关闭时,调用bean的清理方法

    在bean的声明里定义init-method和的story-method,来定义bean的初始化方法和销毁方法。

2.bean的后置处理器

bean的后置处理器允许spring在初始化方法前后,对bean进行额外的处理,其会对IOC容器的所有bean实例进行逐一处理,而非单一实例。例如:对bean实例检查属性的正确性或进行相应的更改。

当你想自定义一个后置处理器时,需要实现Spring提供的BeanPostProcessor接口。

package com.spring.demo.bean;

import org.springframework.beans.BeansException;
import org.springframework.beans.factory.config.BeanPostProcessor; public class MyBeanProcessor implements BeanPostProcessor {
@Override
public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
//添加过滤规则
if ("car".equals(beanName)) {
System.out.println("postProcessBeforeInitialization:" + beanName + bean);
return bean;
}
return null;
} @Override
public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
return bean;
} <!--
配置 bean 后置处理器: 不需要配置 id 属性,
IOC 容器会识别到他是一个 bean 后置处理器, 并调用其方法
-->
<bean class="com.spring.demo.bean.MyBeanProcessor"/>

在这个例子中,我只是在初始化bean之前对bean进行了校验,只是实例化car类型。

当你自定义了一个后置处理器,并将其注册到了Spring容器中,bean的声明周期方法也会发生了一些改变:

  • 通过构造器或工厂方法创建bean的实例;
  • 为bean属性设置值或者引入其他bean;
  • 将bean实例传递给后置处理器的postProcessBeforeInitialization方法;
  • 调用bean的初始化方法,此时bean就可以使用了;
  • 将bean实例传递给后置处理器的postProcessAfterInitialization方法;
  • 容器关闭时,调用bean的清理方法

3.bean的创建方式续

前一篇已经介绍了了通过反射来创建bean的方法,现在介绍剩下的两种:

1.工厂方法

其理念和设计模式中的工厂方法一样。

1.静态工厂方法

静态工厂方法创建bean,不需要创建静态工厂实例,只需调用对应的静态方法即可。

<!--
bean的配置方式2:工厂方法
静态工厂方法:
id指定唯一标识
class指定的是静态工厂方法的类
factory-method指定创建bean的工厂方法
-->
<bean id="carByStaticFactory" class="com.spring.demo.bean.StaticCarFactory" factory-method="getCar">
<constructor-arg value="mini"/>
</bean> //静态工厂
package com.spring.demo.bean; import java.util.HashMap;
import java.util.Map; /**
* 静态方法来创建Car实例
*/
public class StaticCarFactory { private static Map<String, Car> cars = new HashMap<>(10); static {
cars.put("mini", new Car("mini", null, 200, 3000000.00));
cars.put("BMW", new Car("BMW", null, 200, 5000000.00));
} public static Car getCar(String name) {
return cars.get(name);
}
} //测试方法
@Test
public void testStaticFactory() {
ApplicationContext context = new ClassPathXmlApplicationContext("bean-autowire.xml");
Car car = (Car) context.getBean("carByStaticFactory");
System.out.println(car);
}

2.实例工厂方法

与静态工厂方法不同,实例工厂方法需要实例一个工厂对象,其余的与静态工厂方法类似。

<!--
bean的配置方式2:工厂方法
实例态工厂方法:与静态工厂方法类似
id指定唯一标识
class指定的是静态工厂方法的类
factory-bean指定工厂工厂方法实例
factory-method指定创建bean的工厂方法
-->
<bean id="carFactory" class="com.spring.demo.bean.CarFactory"></bean>
<bean id="carByFactory" class="com.spring.demo.bean.Car" factory-bean="carFactory"
factory-method="getCar">
<constructor-arg value="mini"></constructor-arg>
</bean> package com.spring.demo.bean; import java.util.HashMap;
import java.util.Map; public class CarFactory {
private Map<String, Car> cars = new HashMap<>(10); {
cars.put("mini", new Car("mini", null, 200, 3000000.00));
cars.put("BMW", new Car("BMW", null, 200, 5000000.00)); } public Car getCar(String name) {
return cars.get(name);
}
} @Test
public void testFactory() {
ApplicationContext context = new ClassPathXmlApplicationContext("bean-autowire.xml");
Car car = (Car) context.getBean("carByFactory");
System.out.println(car);
}

2.FactoryBean

Spring容器提供了一种新的bean的配置方式:FactoryBean。

Spring容器中有两种Bean:一种是普通的bean,另一种是工厂bean,即FactoryBeean。FactoryBean提供了getObject()来返回bean的实例。当你通过这种方式时,需要实现Spring的FactoryBean接口。

 <!--
bean的配置方式3:FactoryBean
-->
<bean id="carByFacoryBean" class="com.spring.demo.bean.CarFactoryBean"/> package com.spring.demo.bean; import org.springframework.beans.factory.FactoryBean; /**
* 自定义FactoryBean需要实现对应的方法
*/
public class CarFactoryBean implements FactoryBean<Car> { /**
* 该方法返回创建的bean的引用
* @return
* @throws Exception
*/
@Override
public Car getObject() throws Exception {
return new Car("BMW", "mini", 300, 2000000.00);
} /**
* 返回配置的bean的对应的泛型
* @return
*/
@Override
public Class<?> getObjectType() {
return Car.class;
} /**
* 是否单例
* @return
*/
@Override
public boolean isSingleton() {
return true;
}
} @Test
public void testFactory() {
ApplicationContext context = new ClassPathXmlApplicationContext("bean-autowire.xml");
Car car = (Car) context.getBean("carByFactory");
System.out.println(car);
}

3.注解方式

可以采用注解方式来配置bean,不过需要配置主键扫描(其实组件扫描也可以通过注解实现,达到纯注解配置)

组件扫描(component scanning): Spring 能够从 classpath 下自动扫描, 侦测和实例化具有特定注解的组件.

  • @Component: 基本注解, 标识了一个受 Spring 管理的组件:
  • @Respository: 标识持久层组件
  • @Service: 标识服务层(业务层)组件
  • @Controller: 标识表现层组件

    对于扫描到的组件,spring默认使用类名小写来命名,也可以通过value显示指定。
<context:component-scan base-package="com.spring.demo.autowire"/>

@Test
public void testComponentScan() {
ApplicationContext context = new ClassPathXmlApplicationContext("bean-fac.xml");
CarController controller = (CarController) context.getBean("carController");
controller.test();
}

1.组件装配

context:component-scan 元素还会自动注册 AutowiredAnnotationBeanPostProcessor 实例, 该实例可以自动装配具有 @Autowired 和 @Resource 、@Inject注解的属性。

1.使用 @Autowired 自动装配 Bean

@Autowired 注解自动装配具有兼容类型的单个 Bean属性

  • 构造器, 普通字段(即使是非 public), 一切具有参数的方法都可以应用@Authwired 注解
  • 默认情况下, 所有使用 @Authwired 注解的属性都需要被设置. 当 Spring 找不到匹配的 Bean 装配属性时, 会抛出异常, 若某一属性允许不被设置, 可以设置 @Authwired 注解的 required 属性为 false
  • 默认情况下, 当 IOC 容器里存在多个类型兼容的 Bean 时, 通过类型的自动装配将无法工作. 此时可以在 @Qualifier 注解里提供 Bean 的名称. Spring 允许对方法的入参标注 @Qualifiter 已指定注入 Bean 的名称
  • @Authwired 注解也可以应用在数组类型的属性上, 此时 Spring 将会把所有匹配的 Bean 进行自动装配.
  • @Authwired 注解也可以应用在集合属性上, 此时 Spring 读取该集合的类型信息, 然后自动装配所有与之兼容的 Bean.
  • @Authwired 注解用在 java.util.Map 上时, 若该 Map 的键值为 String, 那么 Spring 将自动装配与之 Map 值类型兼容的 Bean, 此时 Bean 的名称作为键值

2.使用 @Resource 或 @Inject 自动装配 Bean

Spring 还支持 @Resource 和 @Inject 注解,这两个注解和 @Autowired 注解的功用类似

  • @Resource 注解要求提供一个 Bean 名称的属性,若该属性为空,则自动采用标注处的变量或方法名作为 Bean 的名称
  • @Inject 和 @Autowired 注解一样也是按类型匹配注入的 Bean, 但没有 reqired 属性

    建议使用 @Autowired 注解

4.bean的作用域

Spring容器的有以下几种作用域:

  • 单例(singleton): 在整个应用中,只创建bean的一个实例。
  • 原型(prototype): 每次注入或者通过Spring应用上下文获取的时候,都会创建一个新的bean的实例。
  • 请求(request): 在web应用中,为每次请求创建一个bean实例。
  • 会话(session): 在web应用中,为每个会话创建一个bean实例。

    在默认情况下,Spring应用上下文中所有bean都是单例的(线程不安全),这在大多数情况下是合理的;但是,有些情况下,你不希望如此。

    比如,电商应用中的购物车,若是单例的,则所有人添加的商品都会添加到一个购物车里面,这就GG了。而现实生活中,你的购物车是只有你能看到并进行更改的。假设有一个Service对象来执行这一业务逻辑:用户登录后添加商品到购物车里面。我们来分析一下:这个Service对象,应该是单例的并且是随着Spring容器加载而实例化好的;但是购物车对象的对象应该是session,并且应该在用户登录后开始向购物车添加商品的时候才进行实例化。而且,Service对象要想完成这一业务逻辑,需要依赖购物车对象。

    简单概况:现在,一个作用域为singleton的Serivce对象需要对一个作用域为session的对象依赖。

    这时,我们可以采取以下方案进行设计:
/**
* 购物车对象,正常应该一个用户创建一次会话的时候就创建一个对象,作用域为session
*/ @Component
@Scope(value = WebApplicationContext.SCOPE_SESSION,
proxyMode = ScopedProxyMode.INTERFACES)
public class ShoppingCart {
} /**
* 购物车对象,正常应该一个用户创建一次会话的时候就创建一个对象,作用域为session
*/ @Component
@Scope(value = WebApplicationContext.SCOPE_SESSION,
proxyMode = ScopedProxyMode.INTERFACES)
public class ShoppingCart {
}

解释一下:Spring并不会将实际的购物车对象注入到service对象中(购物车还没创建,也根本没法注入),而是注入一个购物车对象的代理,这个代理会暴露和购物车一样的方法,所以service对象就会认为他是一个购物车。

而proxyMode属性的值为ScopedProxyMode.INTERFACES,表明这个代理要实现ShoppingCart接口,并将调用委托给bean。

如果shoppingcart是接口的话,这时可以的;但是当其是一个具体类的时候,Spring就不许使用CGlib来生成基于累的代理。

当然,当你采用xml配置文件的形式的时候,只需要设置如下属性即可:

<bean id="cart" class="" scope="session">
<aop:scoped-proxy/>
</bean>

Spiring系列__03IOC补充的更多相关文章

  1. 程序员DD 《Spring boot教程系列》补充

    最近在跟着程序员DD的Spring boot教程系列学习Spring boot,由于年代原因,Spring boot已经发生了一些变化,所以在这里进行一些补充. 补充的知识大多来自评论区,百度,Sta ...

  2. 玩耍Hibernate系列(一)补充--基础知识

    基本概述: Hibernate是一个开放源代码的对象关系映射框架,它对JDBC进行了非常轻量级的对象封装,使得java程序员可以随心所欲的使用对象编程思维来操纵数据库,Hibernate可以应用在任何 ...

  3. iOS逆向系列-tweak补充

    tweak加载资源 开发自己的deb插件需要加载自己的资源,比如图片资源.iOS中常用的两种加载图片资源的方式: + (nullable UIImage *)imageNamed:(NSString ...

  4. ASP.NET MVC5+EF6+EasyUI 后台管理系统(47)-工作流设计-补充

    系列目录 补充一下,有人要表单的代码,这个用代码生成器生成表Flow_Form表的Index代码就可以 加上几个按钮就可以了 <div class="mvctool"> ...

  5. 构建ASP.NET MVC4+EF5+EasyUI+Unity2.x注入的后台管理系统(47)-工作流设计-补充

    原文:构建ASP.NET MVC4+EF5+EasyUI+Unity2.x注入的后台管理系统(47)-工作流设计-补充 系列目录 补充一下,有人要表单的代码,这个用代码生成器生成表Flow_Form表 ...

  6. Android Material Design系列之主题样式介绍说明

    今天这篇文章应该算是Material Design系列的补充篇,因为这篇文章本来应该放到前面讲的,因为讲的是主题嘛,对于一些状态和颜色的介绍,因为我们一新建一个项目时,系统自带了三个属性的颜色,现在就 ...

  7. 带你学够浪:Go语言基础系列-环境配置和 Hello world

    文章每周持续更新,原创不易,「三连」让更多人看到是对我最大的肯定.可以微信搜索公众号「 后端技术学堂 」第一时间阅读(一般比博客早更新一到两篇) 前面几周陆陆续续写了一些后端技术的文章,包括数据库.微 ...

  8. Backbone源码解析(六):观察者模式应用

    卤煮在大概一年前写过backbone的源码分析,里面讲的是对一些backbone框架的方法的讲解.这几天重新看了几遍backbone的源码,才发现之前对于它的理解不够深入,只关注了它的一些部分的细节和 ...

  9. CS Coder学习asp.net5个月的最大感悟:从http的角度重新认识asp.net(二)——我理解的ajax(二)

    啊哈,时隔两个月,才开始写上一篇文章的后续,实在是惭愧.主要是年尾公司又来活了,忙得团团转,而且这段时间在自学mvc.我在上文中,提到过我对mvc框架的初步印象是:相比webform,算是回归了bs本 ...

随机推荐

  1. 在Synology群晖上运行Frp客户端

    一.Synology群晖上开启SSH 二.使用Putty连接 1,登陆管理员账户和密码 2,连接成功后输入sudo su - 输入管理员的密码切换到root权限 下载:wget https://git ...

  2. 第四周WordCount优化

    一.GitHub地址 https://github.com/kawoyi/Advanced-WordCounter最终由组长整合的组长github 二.psp表格 三.个人模块及实现 我负责的是输入模 ...

  3. Codeforces Round #404 (Div. 2) D. Anton and School - 2

    题目链接 转自 给你一个字符串问你能构造多少RSBS. #include<bits/stdc++.h> #define LL long long #define fi first #def ...

  4. 20164305 徐广皓 Exp3 免杀原理与实践

    免杀原理及基础问题回答 实验内容 任务一:正确使用msf编码器,msfvenom生成如jar之类的其他文件,veil-evasion,自己利用shellcode编程等免杀工具或技巧 使用msf编码器生 ...

  5. Linux配置环境变量

    自己mark一下 gedit ~/.bashrc 后面记得要 source ~/.bashrc 使之马上生效(其中波浪线 ~ 代表用户主目录,即home/XX,XX是用户的用户名) Linux下配置环 ...

  6. Python学习笔记-CGI编程(如何在IIS上挂Python开发的Webservice)

    一.如何用Python开发一个简单的Webservice 利用python的cgi编程,可以传入参数将结果输出. 定义需要编码以及需要引用的模块 #conding=utf-8 #修正中文乱码 impo ...

  7. HDU - 1036

    题意描述很垃圾,后来看别人代码才知道怎么回事:对(题目所给d/总时间:所有时间加起来)四舍五入并取整,然后对结果/60得到用了几分钟:对结果%60得到用了几秒. presentation error一 ...

  8. centos7 把终端显示改为英文/中文

    把终端显示改为英文: 1.先备份语言配置文件 cp /etc/locale.conf /home/locale.conf.backup 2.打开配置文件 vim /etc/locale.conf 3. ...

  9. 3种自增ID说明

    自增ID 1.@@identity 所有会话所有表最后一个自增ID 2.IDENT_CURRENT('表名') 所有会话当前表的自增ID 3.SCOPE_IDENTITY() 当前会话所有表最后一个自 ...

  10. 简单的C#网络爬虫

    Source Code: http://download.csdn.net/download/qdalong/10271880 这是爬取网页内容,像是这对大家来说都是不难得,但是在这里有一些小改动,代 ...