1

我试图理解 Spring 中的会话 bean。我正在阅读的那本书说,关于他们:

bean 在需要时创建并存储在 javax.servlet.http.HttpSession 中。当会话被销毁时,bean 实例也被销毁。

我尝试了以下示例:

豆子:

package com.at.test.web;

public class Cart {
    public static int dummy = 0;
    public Cart() {
        System.out.println("Cart::<init> with hashCode " + hashCode());
    }
}

豆定义:

<beans:bean id="cartBean" class="com.at.test.web.Cart" scope="session">
    <apo:scoped-proxy/>
</beans:bean>

控制器:

@Controller
public class HomeController {
    @Autowired 
    private Cart cart;

    @RequestMapping(value = "/", method = RequestMethod.GET)
    public String home(HttpSession session, Model model) {
      System.out.println("Cart is: " + cart.hashCode() 
          + " ; dummy = " + (cart.dummy++) 
          + " (" + cart.getClass().getCanonicalName() + ")" 
          + "; session is: " + session.hashCode());
      return "home.jsp";
  }
}

这是 Tomcat 启动时发生的情况:

Cart::<init> with hashCode 970109301

我认为 Spring 需要这个实例来创建 CGLIB 代理。无论如何,我不确定。

启动后,我用两个不同的浏览器有两个不同的HttpSession。调用控制器时的结果是:

Cart is: 578093288 ; dummy = 0 (com.at.test.web.Cart$$EnhancerByCGLIB$$2eb8334f); session is: 1013723725
Cart is: 578093288 ; dummy = 1 (com.at.test.web.Cart$$EnhancerByCGLIB$$2eb8334f); session is: 1060682497

购物车实例似乎在 HttpSession-s 之间共享,但我期待有两个购物车实例。

如果我让 bean 实现一个接口,使用注释驱动的方法和组件扫描,也是一样的:

public interface ICart {}

--

@Component
@Scope(value="session", proxyMode=ScopedProxyMode.INTERFACES)
public class Cart implements ICart {
    public static int dummy = 0;
    public Cart() {
        System.out.println("Cart::<init> with hashCode " + hashCode());
    }
}

我错过了什么吗?我误解了会话bean的含义吗?

4

2 回答 2

4
public class HomeController {
  @Autowired 
  private Cart cart; <-- Proxy

Cart注入实例的实例HomeController只是将方法调用委托给“真实”实例的代理。类本身还没有自己的Cart方法或状态,所以当然你不会注意到会话之间的任何区别。

于 2013-12-17T17:48:59.883 回答
3

很多代理和委托正在进行。

这个领域

@Autowired 
private Cart cart;

Cart正如您在日志中看到的那样,会话范围在哪里将被代理

com.at.test.web.Cart$$EnhancerByCGLIB$$2eb8334f

但是这个代理没有包装Cart对象。它正在包装 a SimpleBeanTargetSource,它负责CartBeanFactory.

您的 bean 定义附加了一个对象,该SessionScope对象负责HttpSession通过static ThreadLocal. RequestContextHolder当请求从 获取 bean 时BeanFactory,它会将操作委托给SessionScope对象,该对象将检查HttpSession,如果存在则使用那里的 bean,如果不存在则创建并注册一个新的。

您在测试中没有注意到这一点,因为

cart.hashCode()

委托给SimpleBeanTargetSource对象(如果你问我,这是不正确的)。但如果你这样做了

cart.toString();

您会看到实际到达底层Cart对象的差异。


根据您使用的范围和代理策略,所有这些可能会有所不同,但最终目标仍然可以实现。

于 2013-12-17T17:57:18.553 回答