Spring成神之路第三篇:Spring容器基本使用及原理(ApplicationContext接口获取bean的4种方式)
本文主要介绍了spring容器的概念、bean的概念、常见的spring容器,以及spring容器的使用步骤;下一篇我们将详解bean的定义。来源:的位置,容器启动之后会加载这些配文件,然后将这些对象构建好。代码中通过容器提供的getBean方法从容器中获取了HellWorld对象,第一个参数就是xml中bean的id,第二个参数为bean对应的Class对象。本文主要介绍了spring容器的概念
Spring成神之路第三篇:Spring容器基本使用及原理(ApplicationContext接口获取bean的4种方式)
环境
- jdk1.8
- idea
- maven-3.6.1
- spring-5.2.3.RELEASE
IOC容器
IOC容器是具有依赖注入功能的容器,负责对象的实例化、对象的初始化,对象和对象之间依赖关系配置、对象的销毁、对外提供对象的查找等操作,对象的整个生命周期都是由容器来控制。我们需要使用的对象都由ioc容器进行管理,不需要我们再去手动通过new的方式去创建对象,由ioc容器直接帮我们组装好,当我们需要使用的时候直接从ioc容器中直接获取就可以了。
那么spring ioc容器是如何知道需要管理哪些对象呢?
需要我们给ioc容器提供一个配置清单,这个配置支持xml格式和java注解的方式,在配置文件中列出需要让ioc容器管理的对象,以及可以指定让ioc容器如何构建这些对象,当spring容器启动的时候,就会去加载这个配置文件,然后将这些对象给组装好以供外部访问者使用。
这里所说的IOC容器也叫spring容器。
Bean概念
由spring容器管理的对象统称为Bean对象。Bean就是普通的java对象,和我们自己new的对象其实是一样的,只是这些对象是由spring去创建和管理的,我们需要在配置文件中告诉spring容器需要创建哪些bean对象,所以需要先在配置文件中定义好需要创建的bean对象,这些配置统称为bean定义配置元数据信息,spring容器通过读取这些bean配置元数据信息来构建和组装我们需要的对象。
Spring容器使用步骤
- 引入spring相关的maven配置
- 创建bean配置文件,比如bean xml配置文件
- 在bean xml文件中定义好需要spring容器管理的bean对象
- 创建spring容器,并给容器指定需要装载的bean配置文件,当spring容器启动之后,会加载这些配置文件,然后创建好配置文件中定义好的bean对象,将这些对象放在容器中以供使用
- 通过容器提供的方法获取容器中的对象,然后使用
Spring容器对象
spring内部提供了很多表示spring容器的接口和对象,我们来看看比较常见的几个容器接口和具体的实现类。
BeanFactory接口
org.springframework.beans.factory.BeanFactory
spring容器中具有代表性的容器就是BeanFactory接口,这个是spring容器的顶层接口,提供了容器最基本的功能。
常用的几个方法
//按bean的id或者别名查找容器中的bean
Object getBean(String name) throws BeansException
//这个是一个泛型方法,按照bean的id或者别名查找指定类型的bean,返回指定类型的bean对象
<T> T getBean(String name, Class<T> requiredType) throws BeansException;
//返回容器中指定类型的bean对象
<T> T getBean(Class<T> requiredType) throws BeansException;
//获取指定类型bean对象的获取器,这个方法比较特别,以后会专门来讲
<T> ObjectProvider<T> getBeanProvider(Class<T> requiredType);
ApplicationContext接口
org.springframework.context.ApplicationContext
这个接口继承了BeanFactory接口,所以内部包含了BeanFactory所有的功能,并且在其上进行了扩展,增加了很多企业级功能,比如AOP、国际化、事件支持等等。
我们知道可以通过ApplicationContext的getBean方法来获取Spring容器中已初始化的bean。getBean一共有以下四种方法原型:
下来我们分别来探讨以上四种方式获取bean的区别。
(1)getBean(String name)
参数name表示IOC容器中已经实例化的bean的id或者name,且无论是id还是name都要求在IOC容器中是唯一的不能重名。那么这种方法就是通过id或name去查找获取bean.(2)getBean(Class type)
参数Class type表示要加载的Bean的类型。如果该类型没有继承任何父类(Object类除外)和实现接口的话,那么要求该类型的bean在IOC容器中也必须是唯一的。比如applicationContext.xml配置两个类型完全一致的bean,且都没有配置id和name属性我们可以总结getBean(String name)和getBean(Class type)的异同点。
相同点:都要求id或者name或者类型在容器中的唯一性。
不同点:getBean(String name)获得的对象需要类型转换而getBean(Class type)获得的对象无需类型转换。(3)getBean(String name,Class type)
这种方式比较适合当类型不唯一时,再通过id或者name来获取bean。(4)getBean(String name,Object[] args)
这种方式本质还是通过bean的id或者name来获取bean,通过第二个参数Object[] args可以给bean的属性赋值,赋值的方式有两种:构造方法和工厂方法。但是通过这种方式获取的bean必须把scope属性设置为prototype,也就是非单例模式。
下来我们分别来探讨以上四种方式获取bean的区别。
其中实体类Person定义如下:
public class Person {
private String name;
private int age;
public Person(){}
public Person(String name, int age) {
this.name = name;
this.age = age;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
@Override
public String toString() {
return "Person [name=" + name + ", age=" + age + "]";
}
}
applicationContext.xml注册有id为p的bean,配置如下:
<bean id="p" class="com.bean.Person">
<property name="name" value="张三"/>
<property name="age" value="18"/>
</bean>
第一种:l getBean(String name)
参数name表示IOC容器中已经实例化的bean的id或者name,且无论是id还是name都要求在IOC容器中是唯一的不能重名。那么这种方法就是通过id或name去查找获取bean.获取bean的参考代码如下:
@Test
public void testPerson()
{
ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");
Person p = (Person) ctx.getBean("p");
System.out.println(p);
}
第二种:l getBean(Class type)
参数Class type表示要加载的Bean的类型。如果该类型没有继承任何父类(Object类除外)和实现接口的话,那么要求该类型的bean在IOC容器中也必须是唯一的。比如applicationContext.xml配置两个类型完全一致的bean,且都没有配置id和name属性。
<bean class="com.bean.Person">
<property name="name" value="张三"/>
<property name="age" value="18"/>
</bean>
<bean class="com.bean.Person">
<property name="name" value="李四"/>
<property name="age" value="20"/>
</bean>
那么通过com.bean.Person这种类型来查找bean,参考代码如下:
@Test
public void testPerson()
{
ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");
Person p = ctx.getBean(Person.class);
System.out.println(p);
}
但是由于属于com.bean.Person的bean在IOC容器中不唯一,所以这里会抛出NoUniqueBeanDefinitionException异常。
由此我们可以总结getBean(String name)和getBean(Class type)的异同点。
相同点:都要求id或者name或者类型在容器中的唯一性。
不同点:getBean(String name)获得的对象需要类型转换而getBean(Class type)获得的对象无需类型转换。
第三种:l getBean(String name,Class type)
这种方式比较适合当类型不唯一时,再通过id或者name来获取bean。
例如applicationContext.xml配置有如下bean:
<bean id="p1" class="com.bean.Person">
<property name="name" value="张三"/>
<property name="age" value="18"/>
</bean>
<bean name="p2" class="com.bean.Person">
<property name="name" value="李四"/>
<property name="age" value="20"/>
</bean>
参考代码如下:
@Test
public void testPerson()
{
ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");
Person p = ctx.getBean("p2",Person.class);
System.out.println(p);
}
这样可以获取到名字叫”李四”的对象。
第四种:l getBean(String name,Object[] args)
这种方式本质还是通过bean的id或者name来获取bean,通过第二个参数Object[] args可以给bean的属性赋值,赋值的方式有两种:构造方法和工厂方法。但是通过这种方式获取的bean必须把scope属性设置为prototype,也就是非单例模式。
先在com.factory包下设计有如下的工厂类:
public class PersonFactory {
//静态工厂注入
public static Person getPersonInstance(String name,int age)throws Exception
{
Person p = (Person)Class.forName("com.bean.Person").newInstance();
Method m = p.getClass().getMethod("setName", java.lang.String.class);
m.invoke(p, name);
m = p.getClass().getMethod("setAge", int.class);
m.invoke(p, age);
return p;
}
}
在applicationContext.xml中配置有如下bean:
<bean name="p3" class="com.bean.Person" scope="prototype"/>
获取bean的参考代码如下:
@Test
public void testPerson()
{
ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");
Person p = (Person) ctx.getBean("p3",new Object[]{"王五",35});
System.out.println(p);
}
如果想通过工厂注入属性,在applicationContext.xml配置如下bean:
<bean name="p3" class="com.factory.PersonFactory" factory-method="getPersonInstance" scope="prototype">
<constructor-arg name="name">
<null/>
</constructor-arg>
<constructor-arg name="age" value="0"/>
</bean>
ClassPathXmlApplicationContext类
org.springframework.context.support.ClassPathXmlApplicationContext
这个类实现了ApplicationContext接口,注意一下这个类名称包含了ClassPath Xml,说明这个容器类可以从classpath中加载bean xml配置文件,然后创建xml中配置的bean对象,一会后面的案例就会用到这个类。
AnnotationConfigApplicationContext类
org.springframework.context.annotation.AnnotationConfigApplicationContext
这个类也实现了ApplicationContext接口,注意其类名包含了Annotation和config两个单词,上面我们有说过,bean的定义支持xml的方式和注解的方式,当我们使用注解的方式定义bean的时候,就需要用到这个容器来装载了,这个容器内部会解析注解来构建构建和管理需要的bean。
注解的方式相对于xml方式更方便一些,也是我们比较推荐的方式,后面我们会大量使用这种方式,具体会详解。
案例
来个helloworld来详细看一下spring如何使用。
创建项目spring-series
使用idea创建maven项目spring-series,项目坐标:
<groupId>com.javacode2018</groupId>
<artifactId>spring-series</artifactId>
<packaging>pom</packaging>
<version>1.0-SNAPSHOT</version>
spring-series项目中创建一个子模块lesson-001
,项目maven父子结构
spring-series/pom.xml
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>com.javacode2018</groupId>
<artifactId>spring-series</artifactId>
<packaging>pom</packaging>
<version>1.0-SNAPSHOT</version>
<modules>
<module>lesson-001</module>
</modules>
<properties>
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
<!-- 配置maven编译的时候采用的编译器版本 -->
<maven.compiler.compilerVersion>1.8</maven.compiler.compilerVersion>
<!-- 指定源代码是什么版本的,如果源码和这个版本不符将报错,maven中执行编译的时候会用到这个配置,默认是1.5,这个相当于javac命令后面的-source参数 -->
<maven.compiler.source>1.8</maven.compiler.source>
<!-- 该命令用于指定生成的class文件将保证和哪个版本的虚拟机进行兼容,maven中执行编译的时候会用到这个配置,默认是1.5,这个相当于javac命令后面的-target参数 -->
<maven.compiler.target>1.8</maven.compiler.target>
<spring.version>5.2.3.RELEASE</spring.version>
</properties>
<dependencyManagement>
<dependencies>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-core</artifactId>
<version>${spring.version}</version>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-context</artifactId>
<version>${spring.version}</version>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-beans</artifactId>
<version>${spring.version}</version>
</dependency>
</dependencies>
</dependencyManagement>
</project>
目前我们使用spring最新的版本5.2.3.RELEASE
,需要引入spring提供的3个构件
spring-core、spring-context、spring-beans
lesson-001\pom.xml
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<parent>
<artifactId>spring-series</artifactId>
<groupId>com.javacode2018</groupId>
<version>1.0-SNAPSHOT</version>
</parent>
<modelVersion>4.0.0</modelVersion>
<artifactId>lesson-001</artifactId>
<dependencies>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-core</artifactId>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-context</artifactId>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-beans</artifactId>
</dependency>
</dependencies>
</project>
lesson-001中创建HelloWord类
package com.javacode2018.lesson001.demo1;
/**
* 公众号:路人甲Java,工作10年的前阿里P7分享Java、算法、数据库方面的技术干货!坚信用技术改变命运,让家人过上更体面的生活!
*/
public class HelloWorld {
public void say() {
System.out.println("hello,欢迎和【路人甲Java】一起学spring!");
}
}
HelloWord中我们创建了一个say方法,里面会输一段文字。
使用spring容器
下面我们通过spring容器来创建HelloWord对象,并从容器中获取这个对象,然后调用其say方法输出文字。
创建bean xml配置文件
新建一个文件,文件路径如下:
spring-series\lesson-001\src\main\resources\com\javacode2018\lesson001\demo1\bean.xml
bean.xml内容如下:
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-4.3.xsd">
<!--
定义一个bean
id:bean的唯一标识,可以通过这个标识从容器中获取这个bean对象
clss:这个bean的类型,完整类名称
-->
<bean id="helloWorld" class="com.javacode2018.lesson001.demo1.HelloWorld"/>
</beans>
上面就是bean的定义文件,每个xml中可以定义多个bean元素,通过bean元素定义需要spring容器管理的对象,bean元素需指定id和class属性
-
id表示这个bean的标识,在容器中需要唯一,可以通过这个id从容器中获取这个对象;
-
class用来指定这个bean的完整类名
上面的配置文件中我们定义了一个helloWorld
标识的HellWorld类型
的bean对象。
创建测试类
创建一个Client类,如下:
package com.javacode2018.lesson001.demo1;
import org.springframework.context.support.ClassPathXmlApplicationContext;
/**
* 公众号:路人甲Java,工作10年的前阿里P7分享Java、算法、数据库方面的技术干货!坚信用技术改变命运,让家人过上更体面的生活!
*/
public class Client {
public static void main(String[] args) {
//1.bean配置文件位置
String beanXml = "classpath:/com/javacode2018/lesson001/demo1/beans.xml";
//2.创建ClassPathXmlApplicationContext容器,给容器指定需要加载的bean配置文件
ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext(beanXml);
//3.从容器中获取需要的bean
HelloWorld helloWorld = context.getBean("helloWorld", HelloWorld.class);
//4.使用对象
helloWorld.say();
}
}
上面main方法中有容器的详细使用步骤,需要先创建容器对象,创建容器的对象的时候需要指定bean xml文件的位置,容器启动之后会加载这些配文件,然后将这些对象构建好。
代码中通过容器提供的getBean方法从容器中获取了HellWorld对象,第一个参数就是xml中bean的id,第二个参数为bean对应的Class对象。
运行输出
hello,欢迎和【路人甲Java】一起学spring!
总结
本文主要介绍了spring容器的概念、bean的概念、常见的spring容器,以及spring容器的使用步骤;下一篇我们将详解bean的定义。
来源:
https://mp.weixin.qq.com/s?__biz=MzA5MTkxMDQ4MQ==&mid=2648933940&idx=1&sn=6c8c6dc1d8f955663a9874c9f94de88e&chksm=88621e0abf15971c796248e35100c043dac0f5173a870c1d952d4d88a336fa4b76db6885a70c&token=339287021&lang=zh_CN&scene=21#wechat_redirect
的位置,容器启动之后会加载这些配文件,然后将这些对象构建好。
代码中通过容器提供的getBean方法从容器中获取了HellWorld对象,第一个参数就是xml中bean的id,第二个参数为bean对应的Class对象。
运行输出
hello,欢迎和【路人甲Java】一起学spring!
总结
本文主要介绍了spring容器的概念、bean的概念、常见的spring容器,以及spring容器的使用步骤;下一篇我们将详解bean的定义。
来源:
https://mp.weixin.qq.com/s?__biz=MzA5MTkxMDQ4MQ==&mid=2648933940&idx=1&sn=6c8c6dc1d8f955663a9874c9f94de88e&chksm=88621e0abf15971c796248e35100c043dac0f5173a870c1d952d4d88a336fa4b76db6885a70c&token=339287021&lang=zh_CN&scene=21#wechat_redirect
https://blog.csdn.net/qq_23927391/article/details/80625578
更多推荐
所有评论(0)