参考链接
问题
什么是父子容器?
为什么需要用父子容器?
父子容器如何使用?
案例演示
d1包
@Component
public class Service1 {
public String m1() {
return "我是module1中的Servce1中的m1方法";
}
}
@Component
public class Service2 {
@Autowired
private Service1 service1;
public String m1() {
return this.service1.m1();
}
}
@ComponentScan
public class Config {
}
d2包
@Component
public class Service1 {
public String m2() {
return "我是module2中的Servce1中的m2方法";
}
}
@Component
public class Service3 {
@Autowired
private com.example.lurenjia.spring.c23.d2.Service1 service1;
@Autowired
private com.example.lurenjia.spring.c23.d1.Service2 service2;
public String m1() {
return this.service2.m1();
}
public String m2() {
return this.service1.m2();
}
}
@ComponentScan
public class Client {
public static void main(String[] args) {
AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext();
context.register(com.example.lurenjia.spring.c23.d1.Config.class, Client.class);
context.refresh();
}
}
关系梳理,d1包下有三个类,第一个Service1,第二个Service2注入了Service1,配置类扫描包下组件。 d2包下有三个类,第一个Service1不同包同类,第二个Service3,注入了二包的Service1和一包的Service2,第三个是启动类。
输出截图
当容器中存在两个同类型的bean而且我们还不能修改名称的时候就可以考虑使用父子容器。
如何设置父子容器 ?
childContext.setParent(parentContext);
@Override
public void setParent(@Nullable ApplicationContext parent) {
super.setParent(parent);
this.beanFactory.setParentBeanFactory(getInternalParentBeanFactory());
}
测试实例
@ComponentScan
public class Client2 {
public static void main(String[] args) {
AnnotationConfigApplicationContext parentContext = new AnnotationConfigApplicationContext();
parentContext.register(com.example.lurenjia.spring.c23.d1.Config.class);
parentContext.refresh();
AnnotationConfigApplicationContext childContext = new AnnotationConfigApplicationContext();
childContext.register(Client2.class);
childContext.setParent(parentContext);
childContext.refresh();
Service3 service3 = childContext.getBean(Service3.class);
System.out.println(service3.m1());
System.out.println(service3.m2());
}
}
我是module1中的Servce1中的m1方法
我是module2中的Servce1中的m2方法
父子容器的特征
父子容器中的bean
String[] bdn = parentContext.getBeanDefinitionNames();
String[] bdn2 = childContext.getBeanDefinitionNames();
System.out.println(Arrays.toString(bdn));
System.out.println(Arrays.toString(bdn2));
输出
[org.springframework.context.annotation.internalConfigurationAnnotationProcessor,
org.springframework.context.annotation.internalAutowiredAnnotationProcessor,
org.springframework.context.annotation.internalCommonAnnotationProcessor,
org.springframework.context.event.internalEventListenerProcessor,
org.springframework.context.event.internalEventListenerFactory, config, service1, service2]
[org.springframework.context.annotation.internalConfigurationAnnotationProcessor,
org.springframework.context.annotation.internalAutowiredAnnotationProcessor,
org.springframework.context.annotation.internalCommonAnnotationProcessor,
org.springframework.context.event.internalEventListenerProcessor,
org.springframework.context.event.internalEventListenerFactory, client2, service1, service3]
去掉公共部分可以看到 parentContext 有 config, service1, service2,childContext有 client2, service1, service3。
尝试在父容器获取service3,子容器获取service2。
Service1 bean5 = childContext.getBean(Service1.class);
System.out.println(bean5);
com.example.lurenjia.spring.c23.d1.Service1 bean6 = parentContext.getBean(com.example.lurenjia.spring.c23.d1.Service1.class);
System.out.println(bean6);
Service2 bean = childContext.getBean(Service2.class);
System.out.println(bean);
Service2 bean2 = parentContext.getBean(Service2.class);
System.out.println(bean2);
Service3 bean3 = childContext.getBean(Service3.class);
System.out.println(bean3);
Service3 bean4 = parentContext.getBean(Service3.class);
System.out.println(bean4);
输出
com.example.lurenjia.spring.c23.d2.Service1@5ea434c8
com.example.lurenjia.spring.c23.d1.Service1@3bbc39f8
com.example.lurenjia.spring.c23.d1.Service2@4ae3c1cd
com.example.lurenjia.spring.c23.d1.Service2@4ae3c1cd
com.example.lurenjia.spring.c23.d2.Service3@29f69090
Exception in thread "main" org.springframework.beans.factory.NoSuchBeanDefinitionException: No qualifying bean of type 'com.example.lurenjia.spring.c23.d2.Service3' available
at org.springframework.beans.factory.support.DefaultListableBeanFactory.getBean(DefaultListableBeanFactory.java:351)
at org.springframework.beans.factory.support.DefaultListableBeanFactory.getBean(DefaultListableBeanFactory.java:342)
at org.springframework.context.support.AbstractApplicationContext.getBean(AbstractApplicationContext.java:1177)
at com.example.lurenjia.spring.c23.d2.Client2.main(Client2.java:54)
进程已结束,退出代码为 1
总结:
- 使用父子容器后可以解决上述容器中存在两个类型名相同的bean。
- 子容器可以获取到父容器的bean,反之则不可以。
父子容器使用注意点
有的接口方法不支持向上查找,也就是子容器没有的bean不会查找父容器。
例如 getBeanNamesForType
public class Client3 {
public static void main(String[] args) {
DefaultListableBeanFactory parentFactory = new DefaultListableBeanFactory();
parentFactory.registerBeanDefinition("userName",
BeanDefinitionBuilder.
genericBeanDefinition(String.class).
addConstructorArgValue("路人甲Java").
getBeanDefinition());
DefaultListableBeanFactory childFactory = new DefaultListableBeanFactory();
childFactory.setParentBeanFactory(parentFactory);
childFactory.registerBeanDefinition("address",
BeanDefinitionBuilder.
genericBeanDefinition(String.class).
addConstructorArgValue("上海").
getBeanDefinition());
System.out.println("获取bean【userName】:" + childFactory.getBean("userName"));
System.out.println(Arrays.toString(childFactory.getBeanNamesForType(String.class)));
}
}
那么有办法解决这个问题么,使用 BeanFactoryUtils.beanNamesForTypeIncludingAncestors 等带Ancestors这个的方法。
String[] beanNamesForTypeIncludingAncestors = BeanFactoryUtils.beanNamesForTypeIncludingAncestors(childFactory, String.class);
System.out.println(Arrays.asList(beanNamesForTypeIncludingAncestors));
Map<String, String> beansOfTypeIncludingAncestors = BeanFactoryUtils.beansOfTypeIncludingAncestors(childFactory, String.class);
System.out.println(Arrays.asList(beansOfTypeIncludingAncestors));
为什么有的方法会查找父容器有的不会查找父容器?
常见问题
问题1:springmvc中只使用一个容器是否可以?
只使用一个容器是可以正常运行的。
问题2:那么springmvc中为什么需要用到父子容器?
通常我们使用springmvc的时候,采用3层结构,controller层,service层,dao层;父容器中会包含dao层和service层,而子容器中包含的只有controller层;这2个容器组成了父子容器的关系,controller层通常会注入service层的bean。
采用父子容器可以避免有些人在service层去注入controller层的bean,导致整个依赖层次是比较混乱的。
父容器和子容器的需求也是不一样的,比如父容器中需要有事务的支持,会注入一些支持事务的扩展组件,而子容器中controller完全用不到这些,对这些并不关心,子容器中需要注入一下springmvc相关的bean,而这些bean父容器中同样是不会用到的,也是不关心一些东西,将这些相互不关心的东西隔开,可以有效的避免一些不必要的错误,而父子容器加载的速度也会快一些。
|