CometD + Java + jQuery:通过一次调用发布太多

时间:2012-02-16 13:47:25

标签: jquery ajax comet cometd

你好我有下一期使用cometd + jquery + java在广播消息时做一些测试。

这是我的web.xml,我使用了Cometd的Annotations实现:

<!-- Cometd Servlet -->
<servlet>
    <servlet-name>cometd</servlet-name>
    <servlet-class>org.cometd.java.annotation.AnnotationCometdServlet</servlet-class>
    <init-param>
        <param-name>timeout</param-name>
        <param-value>60000</param-value>
    </init-param>
    <init-param>
        <param-name>logLevel</param-name>
        <param-value>3</param-value>
    </init-param>
    <init-param>
        <param-name>services</param-name>
        <param-value>com.api.services.UserStatusService</param-value>
    </init-param>
    <load-on-startup>1</load-on-startup>
    <async-supported>true</async-supported>
</servlet>
<servlet-mapping>
    <servlet-name>cometd</servlet-name>
    <url-pattern>/do/cometd/*</url-pattern>
</servlet-mapping>

这是我注册的服务:

package com.api.services;

import java.util.HashMap;

import javax.inject.Inject;

import org.cometd.bayeux.Message;
import org.cometd.bayeux.client.ClientSession;
import org.cometd.bayeux.client.ClientSessionChannel;
import org.cometd.bayeux.server.BayeuxServer;
import org.cometd.java.annotation.Service;
import org.cometd.java.annotation.Session;
import org.cometd.java.annotation.Subscription;

@Service
public class UserStatusService
{
    @Inject
    private BayeuxServer bayeux;
    @Session
    private ClientSession bayeuxClient;

    @Subscription("/userStatus")
    public void userStatus (Message message)
    {
        HashMap<String, String> mapa = new HashMap<String,String>();
        String channel = message.getChannel();
        System.out.println("*** The Channel is "+channel+" ***");
        ClientSessionChannel chann = bayeuxClient.getChannel(channel);
        mapa.put("canal", channel);
        mapa.put("mensaje", "Hola Wi!!");
        mapa.put("ServChan", bayeux.getChannels().get(0).toString());
        chann.publish(mapa);
    }
}

这是我用于订阅和发布的JS jQuery代码(是一个更大的对象的一部分所以我只是粘贴与此相关的主要内容):

     /** On document ready i call this one **/
     initBroad:function(){
    $.cometd.unregisterTransport('websocket');
    $.cometd.init('http://localhost/MyApp/do/cometd');
    console.log("Set cometd initialization");
    main.broadListener();
},
count:0,
subscription:null,
refresh:function(){
    this.appUnsubscribe();
    this.appSubscribe();
},
appUnsubscribe:function(){
    if (this.subscription) 
        $.cometd.unsubscribe(this.subscription);
    this.subscription = null;
},
appSubscribe:function(){
    func = function(msg){
                    /** I had to do this in order to avoid ALL the 500 times it does it :S **/
        if(main.count < 1){
            console.log(main.count + ": " +  msg.data.mensaje);
        }
        main.count++;
    };
    this.subscription = $.cometd.subscribe("/userStatus",func);
},
broadListener:function(){
    console.log("Set the broadListener");
    main.refresh();
},
publishBroad:function(){
    main.refresh();
    $.cometd.publish('/userStatus', {mensaje:"Hola"});
},

好吧,在尝试通过控制台运行方法publishBroad之后,它实际上运行了,但只需一次点击/请求就可以在服务器上运行450-500次(是的,这只是浏览器向java服务器发出的一个请求,450 -500次只在服务器端重复并在响应中多次到达浏览器)!!

我做错了什么?我使用来自cometd官方网站的latests cometd.js和jquery.cometd.js。

另外,当我在控制台中检查这个(我使用JBoss AS7)时,我留下了一些输出日志行,看看我的调用是否进入了方法(Sys out println表示The Channel是:。它也显示了在JBoss控制台中记录450-500次!

任何人都可以帮我解决这个问题吗?

谢谢!

1 个答案:

答案 0 :(得分:1)

您的问题是您在服务中使用的是ClientSession。

如果在您的服务中,您只想回复发件人,请执行以下操作:

@Service
public class UserStatusService
{
    @Session
    private LocalSession session;

    @Subscribe("/userStatus")
    public void userStatus(ServerSession remoteClient, ServerMessage message)
    {
        Map<String, String> map = new HashMap<String,String>();
        map.put("mensaje", "Hola Wi!!");
        remoteClient.deliver(session, message.getChannel(), map, null);
    }
}

请参阅CometD concepts以了解ClientSession,ServerSession和LocalSession以及annotated services参考之间的区别。

您注意到的循环原因(1个请求导致服务执行500次以上)是因为如果您的服务,您将在最后一行重新广播到同一频道。