精读SpringIoC容器理念及springbeans、context模块关键接口类

360影视 动漫周边 2025-04-25 21:09 2

摘要:在Spring框架中,Bean的实例化和组装都是由IoC容器通过配置元数据完成的。本节主要介绍Spring IoC容器的理念,以及springbeans模块和spring-context模块中的几个关键接口类。

在Spring框架中,Bean的实例化和组装都是由IoC容器通过配置元数据完成的。本节主要介绍Spring IoC容器的理念,以及springbeans模块和spring-context模块中的几个关键接口类。

IoC(Inversion of Control)是“控制反转”的意思。如何理解“控制反转”这个词呢?首先我们需要知道反转的是什么,是由谁来控制。在Spring框架没有出现之前,在Java面向对象的开发中,开发者通过new关键字完成对Object的创建。Spring框架诞生后,是通过Spring容器来管理对象的,因此Object的创建是通过Spring来完成的。最终得出结论:控制反转指的是由开发者来控制创建对象变成了由Spring容器来控制创建对象,创建对象和销毁对象的过程都由Spring来控制。以Spring框架为开发基础的应用尽量不要自己创建对象,应全部交由Spring容器管理。

DI(Dependency Injection)称为依赖注入。在Java程序中,类与类之间的耦合非常频繁,如Class A需要依赖Class B的对象b。而基于Spring框架的开发,在Class A中不需要显式地使用new关键字新建一个对象b,只需在对象b的声明之上加一行注解@Autowired,这样在Class A用到b时,Spring容器会主动完成对象b的创建和注入。这就是Class A依赖Spring容器的注入。通过上面的解释,我们可以发现IoC和DI其实是同一概念从不同角度的解释。

在Spring框架中,org.springframework.context.ApplicationContext接口代表SpringIoC容器,它负责实例化、配置和组装Beans。容器通过读取元数据的配置来获取对象的实例化,以及配置和组装的描述信息。元数据可以用XML、Java注解或Java配置代码表示应用的对象及这些对象之间的内部依赖关系。

Spring框架提供了几个开箱即用的ApplicationContext接口的实现类,如Class-PathXmlApplicationContext、FileSystemXmlApplicationContext和AnnotationConfigApplicationContext等。在独立应用程序中,通常创建一个ClassPathXmlApplication-Context或

FileSystemXmlApplicationContext实例对象来获取XML的配置信息。

开发者也可以指示容器使用Java注解或Java配置作为元数据格式,通过Annotation-ConfigApplicationContext来获取Java配置的Bean。

1. 基于XML的配置

在src/main/resources目录下新建spring.xml文件,内容如上面的代码所示,和标签用来描述Bean的元数据信息。在上面的代码中声明了一个UserService类,该类有两个属性,即id和name,通过

标签直接进行赋值。

UserService实体类的声明代码如下:

//声明UserService类

public class UserService {

private Integer id;

//用户ID

private String name;

//用户名称

//getter和setter方法

public Integer getId {

return id;

}

public void setId(Integer id) {

this.id = id;

}

public String getName {

return name;

}

public void setName(String name) {

this.name = name;

}

//打印属性值

public void getUser {

System.out.println("id:"+this.id);

System.out.println("name:"+this.name);

}

}

以上代码声明了一个UserService类,并实现了属性id和属性name的setter和getter方法,通过getUser方法打印属性值。编写测试代码,展示通过Spring上下文获取UserService对象,具体代码如下:

//测试类

public class SpringXmlTest {

public static void main(String args) {

//通过spring.xml获取Spring应用上下文

ApplicationContext context = new

ClassPathXmlApplication

Context("spring.xml");

UserService userService =

context.getBean("userService",

UserService.class);

userService.getUser;

//打印结果

}

}

打印结果:

在上面的示例代码中,ClassPathXmlApplicationContext可以通过spring.xml文件获取UserService类的配置元数据,通过Spring容器的组装和实例化UserService类,最终正确调用getUser方法打印出定义的属性值。

2. 基于Java注解的配置

从Spring 2.5开始,支持以Java注解的方式来配置Bean,如@Scope、@Service、@Component、@Controller、@Repository、@Autowired和@Resource等注解。

@Scope注解可以设置Bean的作用域。Spring容器实例化的对象默认是单例的,如果想要修改作用域,可以通过@Scope注解进行修改。

表1.1中列出了@Scope注解使用的一些作用域。

表1.1 @Scope注释的作用域

request、session、application和websocket作用域只在Web应用环境中使用。在普通的Spring IoC容器里只有singleton和prototype两种作用域,其他的设置会抛出异常。

下面改造基于XML配置元数据的例子,将其改成基于Java注解的方式来注入Bean,具体代码如下:

//注解的方式声明UserService

@Service

public class UserService {

private Integer id;

//用户ID

private String name;

//用户名称

//getter和setter方法

public Integer getId {

return id;

}

public void setId(Integer id) {

this.id = id;

}

public String getName {

return name;

}

public void setName(String name) {

this.name = name;

}

//属性值打印

public void getUser {

System.out.println("id:"+this.id);

System.out.println("name:"+this.name);

}

}

上面的代码在UserService类中加了一个@Service注解,spring.xml配置文件不再使用。下面增加一个注解类,添加@ComponentScan注解,代码如下:

//@ComponentScan注解用来扫描UserService类

@ComponentScan("com.spring.boot")

public class SpringAnnotationTest {

}

@ComponentScan注解的值是com.spring.boot,说明Spring容器可以自动扫描这个包路径下可管理的类,并对该类进行实例化。添加测试类代码如下:

@ComponentScan("com.spring.boot")

public class SpringAnnotationTest {

public static void main(String args) {

//通过注解类获取应用上下文

ApplicationContext context = new

AnnotationConfigApplication

Context(SpringAnnotationTest.class);

//获取UserService对象

UserService userService =

context.getBean(UserService.class);

userService.setId(1);

userService.setName("zhangsan");

userService.getUser;

//调用方法,打印属性值

}

}

打印结果:

通过AnnotationConfigApplicationContext类可以获取被@Service注解的User-Service实例化对象,并正确打印属性值。通过Java注解的方式同样完成了实例的初始化,说明XML配置方式可以完全被替换。

3. 基于Java配置的示例

从Spring 3.0开始,Spring框架开始支持基于Java的方式来配置元数据,如@Configuration、@Bean、@Import和@Profile等注解。

@Configuration注解一般用来配置类,配置类中可以使用@Bean注解来声明某个类的初始化操作;@Import注解可以导入由@Configuration注解的配置类;@Profile注解可以根据不同环境生成不同的实例。

下面改造基于Java注解的案例,给出一个基于Java配置的示例。

UserService类去掉@Service注解后,将变成普通的Bean。

UserService类的声明代码如下:

//声明UserService类

public class UserService {

private Integer id;

//用户ID

private String name;

//用户名称

public Integer getId {

return id;

}

public void setId(Integer id) {

this.id = id;

}

public String getName {

return name;

}

public void setName(String name) {

this.name = name;

}

//属性值打印

public void getUser {

System.out.println("id:"+this.id);

System.out.println("name:"+this.name);

}

}

新增配置类,代码如下:

//基于@Configuration注解生成UserService对象

@Configuration

public class SpringConfigTest {

@Bean

public UserService userService {

return new UserService;

}

}

SpringConfigTest类由@Configuration注解,表明这个类是个配置类。由@Bean注解的userService方法返回了UserService类的实例。添加测试类代码如下:

@Configuration

public class SpringConfigTest {

@Bean

public UserService userService {

return new UserService;

}

public static void main(String args) {

//通过配置类获取Spring应用上下文

ApplicationContext context = new

AnnotationConfigApplication

Context(SpringConfigTest.class);

UserService userService =

context.getBean(UserService.class);

userService.setId(1);

userService.setName("zhangsan");

userService.getUser;

//打印属性值

}

}

打印结果:

id:1

name:zhangsan

从上面的例子看,基于Java配置实例化对象的方式不再需要对spring.xml的依赖。基于Java注解或Java配置来管理Bean的方式已经是当今编程的流行方式。后文介绍Spring Boot时,还会介绍一些新的注解或配置方式。

如图1.2所示为Bean被Spring容器组装的简单过程。首先通过XML配置、注解配置或Java配置等3种方式配置元数据,然后装配BeanDefinition属性,如果有增强设置,如实现了BeanFactoryPostProcessor或BeanPostProcessor接口,则进行拦截增强处理,最后通过配置的初始化方法完成Bean的实例化。

图1.2 Bean的组装过程

spring-beans模块是Spring容器组装Bean的核心模块,它提供了组装Bean的几个关键接口,如图1.2中的BeanDefinition、BeanFactoryPostProcessor、BeanPost-Processor和BeanFactory等。

BeanDefinition:该接口继承自AttributeAccessor和BeanDefinition两个接口。该接口可以获取Bean的元数据配置信息,也可以改变Bean的属性。

BeanFactoryPostProcessor:该接口为函数接口,只有一个方法postProcessBean-Factory。该接口可以通过ConfigurableListableBeanFactory参数获取Bean-Definition,然后对Bean的属性进行修改,如把Bean的Scope从singleton改为prototype等。

BeanPostProcessor:该接口有两个方法,即postProcessBeforeInitialization和postProcessAfterInitialization,分别用于在Bean实例化之前和实例化之后进行额外的操作。BeanPostProcessor接口与BeanFactoryPostProcessor接口的区别在于,BeanFactoryPostProcessor接口是在Bean实例化之前进行修改。

本节将通过两个简单的例子,展现BeanFactoryPostProcessor和BeanPostProcessor接口的扩展能力。首先来看一个BeanFactoryPostProcessor接口扩展的例子。BeanFactoryPostProcessor接口方法的输入参数是ConfigurableListableBeanFactory,使用该参数可以获取相关Bean的定义信息。示例代码如下:

@Component

public class BeanFactoryPostProcessorImpl implements

BeanFactory

PostProcessor {

@Override

public void

postProcessBeanFactory(ConfigurableListableBean

Factory beanFactory) throws BeansException {

//获取UserService的BeanDefinition

BeanDefinition beanDefinition =

beanFactory.getBeanDefinition

("userService");

//修改Scope属性

beanDefinition.setScope("prototype");

System.out.println(beanDefinition.

getScope);

}

}

打印结果:

通过打印结果可以看到,在UserService实例化之前修改了该类的作用域,将其从singleton改为了prototype。

对于BeanPostProcessor接口的扩展,可以在Spring容器实例化Bean之后或者执行Bean的初始化方法之前添加一些自己的处理逻辑。

示例代码如下:

@Component

public class BeanPostProcessorImpl implements

BeanPostProcessor {

//在实例化之前操作

@Override

public Object postProcessBeforeInitialization(Object

bean, String

beanName) throws BeansException {

//判断Bean的类型

if(bean instanceof UserService){

System.out.println("postProcessBeforeInitialization bean :

" + beanName);

}

return bean;

}

//在实例化之后操作

@Override

public Object postProcessAfterInitialization(Object

bean, String

//判断Bean的类型

System.out.println("postProcessAfterInitialization bean : "

+ beanName);

}

return bean;

}

}

打印结果:

从打印结果中可以看到,在UserService实例化之前和之后都打印了日志,因此通过BeanPostProcessor可以做一些增强逻辑。

来源:大数据架构师

相关推荐