使用DeferredResult进行Spring Long轮询

时间:2015-07-28 17:31:17

标签: spring spring-mvc asynchronous long-polling

我有一个Spring MVC 3.2应用程序,我需要为这个Web服务添加一个Long Polling进行实时聊天。 我遵循了这篇文章Spring MVC 3.2 Preview: Chat Sample

TopicRestController:

 private final Map<DeferredResult<String>, Long> chatRequests =
            new ConcurrentHashMap<DeferredResult<String>, Long>();

 @RequestMapping(value="/{topicId}/updates" , method=RequestMethod.POST)
public @ResponseBody DeferredResult<String> isNewTopic(
        @PathVariable Long topicId,
        Model model, HttpSession session,
        @RequestParam(required = true) String data) throws InterruptedException, CircularDefinitionException{
    logger.info("New long polling request "+topicId);
    final DeferredResult<String> result = new DeferredResult<String>();
    this.chatRequests.put(result, topicId);

    result.onCompletion(new Runnable() {
        @Override
        public void run() {
            chatRequests.remove(result);
            logger.info("Remove request from queue!");
        }
      });

        Timestamp timestamp = new Timestamp(Long.valueOf(data)*1000L);
        String updates = talkService.findNewTopicResponce(topicId,timestamp);
        if (!updates.isEmpty()) {
            result.setResult(updates);
        }

    return result;
}

@RequestMapping(value= "/{categoryId}" + "/addAnswer", method=POST) 
public @ResponseBody Map respTopic(
        @PathVariable Long categoryId,
        @RequestParam String msg,
        @RequestParam(required = false) String imageUrl,
        @RequestParam(required = false) String title,
        @RequestParam long talkId,
        HttpServletRequest request
        ) throws CircularDefinitionException, MessagingException,TalkNotExistException{
     ........................
    for (Entry<DeferredResult<String>, Long> entry :  this.chatRequests.entrySet()){
                            if(entry.getValue().equals(talkId)){        
                                entry.getKey().setResult(""+talkId);
                            }
                        }

    }

现在的问题是:

当我调用“/ {topicId} / updates”时如果30秒后没有任何答案服务器返回错误500,如果有人写了一条消息,服务器返回正确的消息,但服务器总是在30秒后响应,我需要服务器在有人写新消息时立即响应而不是在超时过程中响应。

1 个答案:

答案 0 :(得分:0)

我遇到了同样的问题,我遇到了类似的问题:Tomcat 7 server error after 10 seconds when long polling

基本上,你必须告诉Tomcat不要超时。您可以通过编辑server.xml Tomcat文件(在conf目录中)来包含asyncTimeout参数来执行此操作。这会告诉你的Tomcat在一分钟后超时:

<Connector port="8080" protocol="HTTP/1.1" asyncTimeout="60000" connectionTimeout="20000" redirectPort="8443" />

使用值-1表示Tomcat永远不会超时。