spring同名bean覆盖问题
默认情况下,spring在处理同一个ApplicationContext中名称相同的bean时
分为两种情况处理
1、如果两个bean是在同一个配置文件中,那么spring会报错。
2、如果两个bean是在不同的配置文件中,默认情况下,spring会覆盖先前的bean。
在配置文件很多时,如果在启动时,对于同名的bean加载没有异常信息,出现问题后会比较难以定位。
在spring中,处理容器的元数据信息时,默认使用DefaultListableBeanFactory类,该类中有个属性:allowBeanDefinitionOverriding,默认情况下为true,即允许重名的bean可以被覆盖。
还好,spring有办法对改属性赋值。
重写ContextLoaderListener,对于web应用,容器类型为XmlWebApplicationContext,在该类中设置allowBeanDefinitionOverriding为false,然后在spring启动时,碰到同名bean就会抛出异常。
案例如下
1
2
3
4
5
6
7
8
9
|
public class TradeContextLoaderListener extends ContextLoaderListener { @Override protected void customizeContext(ServletContext servletContext, ConfigurableWebApplicationContext applicationContext) { super .customizeContext(servletContext, applicationContext); XmlWebApplicationContext context = (XmlWebApplicationContext) applicationContext; context.setAllowBeanDefinitionOverriding( false ); } } |
配置web.xml:
1
2
3
4
|
< listener > < description >spring监听器</ description > < listener-class >com.***.trade.system.web.util.TradeContextLoaderListener</ listener-class > </ listener > |
spring 子类覆盖父类中注入的bean
我们在设计程序框架的时候,会设计一个抽象基类,子类继承这个基类,共有的方法放到基类中去,使用spring后使代码变的很简单,现在遇到的问题是在基类中注入bean后,子类不可能都会是有这个bean,那么需要考虑到子类需要覆盖或者说重新注入个性化的bean
有三种方法来实现这个效果,以下是一种方法,如下面代码:
抽象基类
1
2
3
4
|
public abstract class AbstractNameService { public abstract String getname(); } |
两个实现类:
1
2
3
4
5
6
7
8
9
|
@Service ( "firstNameService" ) public class FirstNameService extends AbstractNameService { @Override public String getname() { return "FirstName" ; } } |
1
2
3
4
5
6
7
8
9
|
@Service ( "nameService" ) public class NameService extends AbstractNameService { @Override public String getname() { return "Name" ; } } |
另外一个抽象基类
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
|
public abstract class AbstractService { protected AbstractNameService nameService; public String getName() { return nameService.getname(); } public AbstractNameService getService() { return nameService; } <span style= "color:#ff9966;" > @Resource (name = "nameService" )</span> public void setService(AbstractNameService nameService) { this .nameService = nameService; } } |
实现类:
1
2
3
4
5
6
7
8
9
10
|
@Service ( "getNameService" ) public class GetNameService extends AbstractService { <span style= "color:#ff9900;" > @Resource (name = "firstNameService" )</span> @Override public void setService(AbstractNameService nameService) { this .nameService = nameService; } } |
controller
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
|
@Controller public class UnionpayQuickPayDSMVC { @Resource private AbstractService getNameService; @RequestMapping (value = "/*" , method = RequestMethod.GET) public void execute(HttpServletRequest request, HttpServletResponse response) { try { response.getWriter().write(getNameService.getName()); } catch (IOException e) { System.out.println(e); } } } |
在applicationContext.xml和springmvc的配置文件只需要添加一个包<context:component-scan/>标签就行了
以上为个人经验,希望能给大家一个参考,也希望大家多多支持服务器之家。
原文链接:https://blog.csdn.net/ado1986/article/details/49334791