二级接口ListableBeanFactory
package org.springframework.beans.factory; import java.lang.annotation.Annotation;
import java.util.Map; import org.springframework.beans.BeansException;
import org.springframework.core.ResolvableType; //可将Bean逐一列出的工厂
public interface ListableBeanFactory extends BeanFactory { //是否存在给定的bean名称的信息
boolean containsBeanDefinition(String beanName); //BeanDefinition的总数
int getBeanDefinitionCount(); //返回工程中所有bean的名字
String[] getBeanDefinitionNames(); //返回对于指定类型的beanName
String[] getBeanNamesForType(ResolvableType type); //返回指定类型bean的名字
String[] getBeanNamesForType(Class<?> type); /*
* 返回指定类型的名字
* includeNonSingletons为false表示只取单例Bean,true则不是
* allowEagerInit为true表示立刻加载,false表示延迟加载。
* 注意:FactoryBeans都是立刻加载的。
*/
String[] getBeanNamesForType(Class<?> type, boolean includeNonSingletons, boolean allowEagerInit); //根据bean的类型返回bean和bean的Map集合
<T> Map<String, T> getBeansOfType(Class<T> type) throws BeansException; <T> Map<String, T> getBeansOfType(Class<T> type, boolean includeNonSingletons, boolean allowEagerInit)
throws BeansException; // 根据注解类型,查找所有有这个注解的Bean名和Bean的Map
String[] getBeanNamesForAnnotation(Class<? extends Annotation> annotationType); // 根据注解类型,查找所有有这个注解的Bean名和Bean的Map
Map<String, Object> getBeansWithAnnotation(Class<? extends Annotation> annotationType) throws BeansException; //根据指定Bean名和注解类型查找指定的Bean
<A extends Annotation> A findAnnotationOnBean(String beanName, Class<A> annotationType)
throws NoSuchBeanDefinitionException; }
这个工厂中扩展了
3个 BeanDefinition有关的接口,
3个根据指定类型返回beanName的数组
2个根据指定类型返回bean和存储Bean的Map集合
3个和注解有关的返回bean和beanMap的接口方法
对于BeanDefinition中包含了一个类在Spring工厂中所有属性。
这个工厂如同名字一样,可以返回所有bean的实例,但是这个工厂并没有直接返回实例的方法,只有通过条件返回指定bean的Name数组,Map等,
其实不需要直接返回实例的方法,因为我们拿到了bean的所有名字,我们就可以使用getbean,父接口中得方法,也就是BeanFactory进行获取Bean的实例!
可以通过ApplicationContext的getBean方法来获取Spring容器中已初始化的bean。getBean一共有以下四种方法原型:
getBean(String name) getBean(Class<T> type) getBean(String name,Class<T> type) getBean(String name,Object[] args)
下来我们分别来探讨以上四种方式获取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>
第一种方法: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); }
第二种方法:getBean(Class<T> type)
参数Class<T> 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<T> type)的异同点。
相同点:都要求id或者name或者类型在容器中的唯一性。
不同点:getBean(String name)获得的对象需要类型转换而getBean(Class<T> type)获得的对象无需类型转换。
第三种方法:getBean(String name,Class<T> type)
这种方式比较适合当类型不唯一时,再通过id或者name来获取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);
}
这样可以获取到名字叫”李四”的对象。
第四种方法: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>
---------------------
参考链接:
https://www.cnblogs.com/java-synchronized/p/6777556.html
https://blog.csdn.net/qq_23927391/article/details/80625578
二级接口ListableBeanFactory的更多相关文章
- 2.二级接口ListableBeanFactory
这个随笔主要讲的是ListableBeanFactory package org.springframework.beans.factory; import java.lang.annotation. ...
- 3.二级接口HierarchicalBeanFactory
HierarchicalBeanFactory 字面意思是分层工厂, 那么这个工厂是怎么分层的呢? package org.springframework.beans.factory; //分层工 ...
- Spring源码分析——BeanFactory体系之接口详细分析
Spring的BeanFactory的继承体系堪称经典.这是众所周知的!作为Java程序员,不能错过! 前面的博文分析了Spring的Resource资源类Resouce.今天开始分析Spring的I ...
- Spring之2:HierarchicalBeanFactory接口
HierarchicalBeanFactory:HierarchicalBeanFactory继承BeanFactory并扩展使其支持层级结构.getParentBeanFactory()方法或者父级 ...
- Spring源码学习之BeanFactory体系结构
一.BeanFactory BeanFactory是Spring IOC容器的鼻祖,是IOC容器的基础接口,所有的容器都是从它这里继承实现而来.可见其地位.BeanFactory提供了最基本的IOC容 ...
- Spring 之 BeanFactory 源码 - 接口分析
一.BeanFactory的基本类体系结构(接口为主):
- Spring系列之beanFactory与ApplicationContext
一.BeanFactoryBeanFactory 是 Spring 的“心脏”.它就是 Spring IoC 容器的真面目.Spring 使用 BeanFactory 来实例化.配置和管理 Bean. ...
- Spring的Bean,AOP以及工具类初探
1.Bean(Ioc) BeanWrapper 根据JavaDoc中的说明,BeanWrapper提供了设置和获取属性值(单个的或者是批量的),获取属性描述信息.查询只读或者可写属性等功能.不仅如此, ...
- Spring源码解析一:IOC容器设计
一.IOC接口设计 IOC容器设计的源码主要在spring-beans.jar.spring-context.jar这两个包中.IOC容器主要接口设计如下: 这里的接口设计有两条主线:BeanFact ...
随机推荐
- .Net Core 项目在Windows服务中托管【转载】
本文以创建的WebAPI项目为例子进行讲解(本人使用VS Code创建的项目) 1.使用VS Code创建WebAPI项目(项目名称自定义) 2.在创建的项目csproj项目文件中,确认是否存在运行时 ...
- Eclipse Build path
Build Path用于设置Java的构建路径,管理Java工程所包含的资源,使工程结构清晰合理. 包括以下几项: Source Source包括 source folder和output folde ...
- Hanlp分词1.7版本在Spark中分布式使用记录
新发布1.7.0版本的hanlp自然语言处理工具包差不多已经有半年时间了,最近也是一直在整理这个新版本hanlp分词工具的相关内容.不过按照当前的整理进度,还需要一段时间再给大家详细分享整理的内容.昨 ...
- mongodb集群配置分片集群
测试环境 操作系统:CentOS 7.2 最小化安装 主服务器IP地址:192.168.197.21 mongo01 从服务器IP地址:192.168.197.22 mongo02 从服务器IP地址: ...
- maya中MFnMesh.h使用说明的翻译
由于最近要修改一个maya中的deformer脚本,于是开始系统学习openMaya的一些知识,当然少不了得把一堆头文件说明看一遍.首先把MFnMesh.h这个文件翻译一下吧,不废话,上译文: 首先M ...
- uCOS-III等RTOS与IoT OS
一.IoT 以百度天工物联网为例,如下图所示: 通过联网设备,将设备连接至云端,并将每个设备的信息进行上传,并在云端进行设备的管理,设备数据的处理计算.存储,可视化的展示和分析.IoT设备较传统的嵌入 ...
- [UE4]Menu Anchor,菜单锚点
一.想要弹出某个菜单的时候,Menu Anchor可以做为菜单弹出的位置. 二.Menu Anchor本身不显示任何东西 三.Menu Class:选择要弹出的UI,可以是任意的UserWidget ...
- C# 语言历史版本特性(C# 1.0到C# 7.1汇总更新)
历史版本C#作为微软2000年以后.NET平台开发的当家语言,发展至今具有17年的历史,语言本身具有丰富的特性,微软对其更新支持也十分支持.微软将C#提交给标准组织ECMA,C# 5.0目前是ECMA ...
- [SQL]sql server中如何直接查询存储过程EXEC返回的结果集?
Declare @T Table (iDay VARCHAR(),iNum DECIMAL(,),yuxiang DECIMAL(,)) Insert @T --EXEC [dbo].[BSP0101 ...
- BZOJ 3097: Hash Killer I
3097: Hash Killer I Time Limit: 5 Sec Memory Limit: 128 MBSec Special Judge[Submit][Status][Discus ...