@Scope("prototype") bean 范围不创建新 bean

新手上路,请多包涵

我想在我的控制器中使用带注释的原型 bean。但是 spring 正在创建一个单例 bean。下面是代码:

 @Component
@Scope("prototype")
public class LoginAction {

  private int counter;

  public LoginAction(){
    System.out.println(" counter is:" + counter);
  }
  public String getStr() {
    return " counter is:"+(++counter);
  }
}

控制器代码:

 @Controller
public class HomeController {
    @Autowired
    private LoginAction loginAction;

    @RequestMapping(value="/view", method=RequestMethod.GET)
    public ModelAndView display(HttpServletRequest req){
        ModelAndView mav = new ModelAndView("home");
        mav.addObject("loginAction", loginAction);
        return mav;
    }

    public void setLoginAction(LoginAction loginAction) {
        this.loginAction = loginAction;
    }

    public LoginAction getLoginAction() {
        return loginAction;
    }
    }

速度模板:

  LoginAction counter: ${loginAction.str}

Spring config.xml 启用了组件扫描:

     <context:annotation-config />
    <context:component-scan base-package="com.springheat" />
    <mvc:annotation-driven />

我每次都得到一个递增的计数。不知道我哪里错了!

更新

正如 @gkamal 所建议的,我做了 HomeController webApplicationContext 并解决了问题。

更新代码:

 @Controller
public class HomeController {

    @Autowired
    private WebApplicationContext context;

    @RequestMapping(value="/view", method=RequestMethod.GET)
    public ModelAndView display(HttpServletRequest req){
        ModelAndView mav = new ModelAndView("home");
        mav.addObject("loginAction", getLoginAction());
        return mav;
    }

    public LoginAction getLoginAction() {
        return (LoginAction) context.getBean("loginAction");
    }
}

原文由 tintin 发布,翻译遵循 CC BY-SA 4.0 许可协议

阅读 668
2 个回答

作用域原型意味着每次你向 spring(getBean 或依赖注入)请求一个实例时,它都会创建一个新实例并给出一个引用。

在您的示例中,创建了一个新的 LoginAction 实例并将其注入到您的 HomeController 中。如果您有另一个控制器,您将 LoginAction 注入其中,您将获得一个不同的实例。

如果您希望每次调用都有不同的实例 - 那么您每次都需要调用 getBean - 注入单例 bean 将无法实现。

原文由 gkamal 发布,翻译遵循 CC BY-SA 3.0 许可协议

自 Spring 2.5 以来,有一种非常简单(且优雅)的方法可以实现这一点。

您只需更改 @Scope 注释的参数 proxyModevalue 即可。

使用这个技巧,您可以避免每次在单例 bean 中需要原型时编写额外的代码或注入 ApplicationContext。

例子:

 @Service
@Scope(value="prototype", proxyMode=ScopedProxyMode.TARGET_CLASS)
public class LoginAction {}

使用上面的配置 LoginAction (内部 HomeController )始终是 _原型_,即使控制器是 _单例_。

原文由 db80 发布,翻译遵循 CC BY-SA 4.0 许可协议

撰写回答
你尚未登录,登录后可以
  • 和开发者交流问题的细节
  • 关注并接收问题和回答的更新提醒
  • 参与内容的编辑和改进,让解决方法与时俱进
推荐问题