9

我们目前有一个注入到 Servlet 中的 Stateful bean。问题是有时我们Caused by: javax.ejb.ConcurrentAccessException: SessionBean is executing another request. [session-key: 7d90c02200a81f-752fe1cd-1]在有状态 bean 上执行方法时会遇到错误。

public class NewServlet extends HttpServlet {  
    @EJB  
    private ReportLocal reportBean;

    protected void processRequest(HttpServletRequest request, HttpServletResponse response)
    throws ServletException, IOException {
        response.setContentType("text/html;charset=UTF-8");
        PrintWriter out = response.getWriter();
        try {
           String[] parameters  = fetchParameters(request);
           out.write(reportBean.constructReport(parameters));
        } finally { 
            out.close();
        }
    } 
}

在上面的代码中,constructReport将检查它是否需要打开到报告中指定的数据库的新连接,然后根据指定参数构建的查询构建 HTML 格式的报告。

我们之所以选择使用有状态 bean 而不是无状态 bean,是因为我们需要打开与未知数据库的数据库连接并对其执行查询。对于无状态 bean,重复打开和关闭与每个注入的 bean 实例的数据库连接似乎非常低效。

4

5 回答 5

14

有关ConcurrentAccessException的更多详细信息:根据 EJB 规范,对 SLSB 的访问由应用程序同步。服务器。但是,SFSB 并非如此。确保 SFSB 不被同时访问的负担落在了应用程序开发人员的肩上。

为什么?好吧,SLSB 的同步仅在实例级别是必需的。也就是说,SLSB 的每个特定实例都是同步的,但是您可能在一个池中或集群中的不同节点上有多个实例,不同实例上的并发请求不是问题。不幸的是,对于 SFSB,这并不容易,因为实例的钝化/激活以及跨集群的复制。这就是规范不强制执行此操作的原因。如果您对该主题感兴趣,请查看此讨论。

这意味着从 servlet 使用 SFSB 很复杂。具有来自同一会话的多个窗口的用户,或在渲染完成之前重新加载页面可能导致并发访问。理论上,在 servlet 中完成的对 EJB 的每次访问都需要在 bean 本身上同步。我所做的是创建一个InvocationHandler来同步特定 EJB 实例上的所有调用:

public class SynchronizationHandler implements InvocationHandler {

 private Object target;  // the EJB

 public SynchronizationHandler( Object bean )
 {
        target = bean;
 }

  public Object invoke(Object proxy, Method method, Object[] args) throws Throwable
  {
    synchronized( target )
    {
       // invoke method to the target EJB
    }
  }

}

然后,在您获得对 EJB 的远程引用之后,您使用SynchronizationHandler. 这样您就可以确保不会从您的应用程序同时访问此特定实例(只要它仅在一个 JVM 中运行)。您还可以编写一个常规的包装类来同步 bean 的所有方法。

尽管如此,我的结论是:尽可能使用 SLSB。

编辑

这个答案反映了 EJB 3.0 规范(第 4.3.13 节):

不允许客户端对有状态会话对象进行并发调用。如果客户端调用的业务方法正在一个实例上进行,而另一个客户端调用的调用(来自相同或不同的客户端)到达有状态会话 bean 类的同一实例,如果第二个客户端是 bean 业务的客户端接口,并发调用可能会导致第二个客户端收到 javax.ejb.ConcurrentAccessException

EJB 3.1 中删除了此类限制(第 4.3.13 节):

默认情况下,允许客户端对有状态会话对象进行并发调用,并且容器需要序列化此类并发请求。

[...]

Bean 开发人员可以选择指定禁止对有状态会话 bean 的并发客户端请求。这是使用 @AccessTimeout 注释或值为 0 的 access-timeout 部署描述符元素完成的。在这种情况下,如果客户端调用的业务方法在另一个客户端调用的实例上正在进行中,则来自相同或不同的调用客户端,到达有状态会话 bean 的同一实例,如果第二个客户端是 bean 的业务接口或无接口视图的客户端,则并发调用必须导致第二个客户端接收 javax.ejb.ConcurrentAccessException

于 2009-12-20T10:29:45.717 回答
9

这不是有状态会话 bean (SFSB) 的用途。它们被设计为保存会话状态,并且将绑定到用户的 http 会话以保存该状态,就像直接在会话中存储状态的重量级替代方案一样。

如果你想保存数据库连接之类的东西,那么有更好的方法来解决它。

最好的选择是使用连接池。您应该始终使用连接池,并且如果您在应用程序服务器中运行(如果您使用的是 EJB,那么您就是这样),那么您可以轻松地使用应用服务器的数据源配置来创建连接池,并使用在您的无状态会话 bean (SLSB) 中。

于 2009-12-20T09:32:50.960 回答
1

在您提供一些代码和堆栈跟踪之前,我建议您考虑使用连接池。如果“未知数据库”是指其参数由最终用户提供的数据库,因此不可能预先配置连接池,您仍然可以使用连接池概念,而不是每次都打开一个新连接。

另外,请检查此线程

于 2009-12-20T08:54:11.240 回答
0

会话 bean 不能同时使用,就像 skaffman 所说的那样,它们旨在处理与客户端会话对应的状态,并且通常存储在每个客户端的会话对象中。

重构以使用数据库池来处理对资源的并发请求是可行的方法。

同时,如果您只需要这种微不足道的使用,您可以同步对constructReport的调用,如下所示:

synchronised (reportBean) {
       out.write(reportBean.constructReport(parameters));
}

请注意,如果相对于您的客户数量来说,constructReport 花费大量时间,这不是解决方案。

于 2009-12-20T09:48:53.313 回答
0

你永远不应该同步 servlet 或 ejb 访问,因为这会导致请求队列,如果你有 N 个并发用户,最后一个用户将等待很长时间并且经常得到超时响应!!!同步方法不是出于这个原因!!!

于 2011-07-14T07:19:29.980 回答