基于spring同名bean覆盖问题的解决

时间:2022-05-20 02:17:01

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