如何确定JMS连接是否存在?

时间:2010-07-06 15:56:49

标签: java jms

在JMS中,很容易发现连接是否丢失,发生异常。但是,如何确定连接是否再次存在?

场景:我使用JMS与我的服务器通信。现在我的连接断开(服务器关闭),这导致异常。到现在为止还挺好。如果服务器再次启动并重新建立连接,我该如何知道?

我没有看到任何能够提供此类信息的听众。

4 个答案:

答案 0 :(得分:16)

啊......旧的异常处理/重新连接难题。

有些传输提供商会自动为您重新连接您的应用程序,以及一些使应用程序驱动器重新连接的应用程序。通常,重新连接会将异常隐藏在应用程序之外。不利的一面是,如果所有远程消息传递节点都已关闭,您不希望应用程序永久挂起,因此最终,必须包含一些重新连接逻辑。

现在,这是有趣的部分 - 您如何以提供商中立的方式处理异常? JMS例外实际上毫无价值。例如,“安全性异常”可能是Java安全策略过于严格,文件系统权限过于严格,LDAP凭据失败,传输连接失败,队列或主题打开失败或其他几十个与安全相关的问题。这是链接的异常,它具有来自传输提供程序的详细信息,可以帮助调试问题。我的客户一般采取三种不同方法中的一种......

  1. 同等对待所有错误。关闭所有对象并重新初始化它们。这是JMS portable。
  2. 允许应用检查链接的异常,以区分致命错误和瞬态错误(即auth错误与队列已满)。不是提供者便携式。
  3. 特定于提供程序的错误处理类。另外两个的混合。
  4. 在您的情况下,队列和主题对象可能仅在原始连接的上下文中有效。假设提供者自动重新连接您获得异常的事实意味着重新连接失败,并且无法恢复队列和主题对象的上下文。关闭所有对象并重新连接。

    您是否想要做一些更具特定于提供者的事情,例如区分瞬态和永久性错误,这是“依赖”的事情之一,您必须根据具体情况来解决这个问题。

答案 1 :(得分:6)

监视连接异常的最佳方法是设置异常监听器,例如:

ConnectionFactory connectionFactory = (ConnectionFactory) context.lookup("jmsContextName");
        connection = connectionFactory.createConnection();
        connection.setExceptionListener(new ExceptionListener() {
            @Override
            public void onException(JMSException exception) {
                logger.error("ExceptionListener triggered: " + exception.getMessage(), exception);
                try {
                    Thread.sleep(5000); // Wait 5 seconds (JMS server restarted?)
                    restartJSMConnection();
                } catch (InterruptedException e) {
                    logger.error("Error pausing thread" + e.getMessage());
                }
            }
        });
        connection.start();

答案 2 :(得分:4)

JMS规范没有描述任何传输协议,它没有说任何关于连接的事情(即代理应该保持它们存活或为每个会话建立新的连接)。所以,我认为你的意思是

  

现在我的连接中断(服务器已关闭),这会导致异常。

是您尝试发送消息并且您收到JmsException。

我认为,查看经纪人是否正常的唯一方法是尝试发送消息。

答案 3 :(得分:1)

对于基于连接的JMSException,您唯一的选择是尝试在异常处理程序中重新建立连接,然后重试该操作。