3

据我所知,Spring bean 默认是单例的。

我想要的是考虑实例属性使 bean 成为线程安全的。我将尝试使用一个简单的示例向您展示。
考虑以下代码:

@Controller
public class MyServlet {

    @Autowired
    private HelloService service;

    @RequestMapping(value="/hello", method = RequestMethod.GET)
    public void sayHello(HttpServletRequest req, HttpServletResponse res) throws IOException {
        service.doStuff();
    }

}


public class HelloService {

    private int i = 1;

    public void doStuff() {
        System.out.println("Started " + i);
        i++;
        System.out.println(Thread.currentThread().getName() + " Done " + i);
    }
}

输出将是这样的:

32911580@qtp-28064776-0 - Started 1
7802158@qtp-28064776-2 - Started 2
32911580@qtp-28064776-0 - Done 3
7802158@qtp-28064776-2 - Done 3

这证明了“i”变量在多个线程之间共享。

我还尝试将 HelloService bean 定义为原型,就像这样

<bean id="helloService" class="my.package.HelloService" scope="prototype" />

但结果是一样的。

我发现解决这个问题的唯一方法是: - 将声明移动到 doStuff() 方法中,但这不是我想要的 - 制作 doStuff() 方法,但这意味着有锁

我想要的是在每次通话时都有一个新的 HelloService 实例。

谁能帮我?提前致谢。

更新

我使用查找方法找到了方法注入的解决方案。 http://static.springsource.org/spring/docs/3.1.1.RELEASE/spring-framework-reference/html/beans.html#beans-factory-lookup-method-injection

4

3 回答 3

3

由于您只有一个 MyServlet 实例,因此您也将只有一个 HelloService 实例。

以下 spring 范围之一有帮助: request 每个 HTTP 请求的新实例。 session 在 servlet 容器中创建的每个新 HttpSession 的类的新实例。

这将改变代码的语义。如果您在范围请求中声明 bean,您的计数器将始终为 1。

要拥有一个没有锁的唯一计数器,您可以使用 java.util.concurrent.atomic.AtomicInteger。

另一方面,锁没有什么不好的。除极端情况外,在大多数情况下,性能影响可以忽略不计。

构建应用程序后,请使用http://vmlens.com检查竞争条件。

于 2013-10-24T07:02:56.907 回答
1

我也尝试将 HelloService bean 定义为原型,但结果是一样的。

你得到了同样的结果,因为你MyServlet是单身。所以你只有一个MyServlet实例,因此你只有一个HelloService实例。如果您也制作MyServlet原型bean,则每次都会获得两者的新实例

于 2017-04-11T12:06:26.073 回答
1

添加到 Thomas 的答案中,对于服务 bean,请求范围是有状态和线程安全组件的最佳选择,因为在该范围内通常只调用一个实例。

如果在特定应用程序范围内需要多个(有状态和线程安全)bean,则原型范围将是最合适的选择。

于 2017-03-26T21:38:09.513 回答