comet4j

初步学习可以看看
java web 服务器推送技术--comet4j
Comet HTTP服务器推送框架之Comet4J
java 使用comet4j向客户端主动推送例子

比较完善的应用
comet4j运用-- 聊天室例子
comet4j运用-- 间隔时间推送服务器剩余内存大小

几篇不错的文章
使用 Java 实现 Comet 风格的 Web 应用
浅析Comet技术在Java Web实时系统开发中的应用
java web 服务器推送技术 comet实现(原理)

转载自comet4j,开发文档被墙了,就直接贴过来方便查看
Tomcat 8 测试失败,不知啥时更新(有成功的望)
注:和Tomcat7 相同的配置,服务器端不报错,但是客户端会报“JSON 解析错误”,404的错误

简介

Comet4J是一个微型的即时推送框架,它分为服务端与客户端两部分,你只要将服务器端(JAR文件,目前仅支持Tomcat6、7)放入WEB-INF\lib,客户端(JavaScript文件)引入到页面,那么你的应用就具备了向客户端推送信息的能力,而你仅需要在服务器端调用Comet4J所提供发送方法,信息就会被主动的推送到客户的浏览器上。

准备工作

  • 下载服务端jar文件

Comet4J目前仅支持Tomcat6、7版本,根据您所使用的Tomcat版本下载【comet4j-tomcat6.jar】或【comet4j-tomcat7.jar】文件放置到WEB项目的WEB-INF\lib目录下。

  • 下载客户端js文件

    下载【comet4j.js】到您的项目中,比如:WebContent\js目录下。

  • 修改服务器配置文件

    因为Comet4J工作在NIO方式下,所以我们需要调整服务器连接器配置,更换为NOI连接器。 打开server.xml文件将找到原先的连接器配置:
    <Connector executor="tomcatThreadPool" port="8080" protocol="HTTP/1.1" connectionTimeout="20000" redirectPort="8443" />
    替换为:
    <Connector URIEncoding="UTF-8" connectionTimeout="20000" port="8080" protocol="org.apache.coyote.http11.Http11NioProtocol" redirectPort="8443"/>

  • 在web.xml中加载Comet4J框架

    最后我们需要在web.xml配置侦听和comet连接地址,以使Comet4J生效:

 <listener> 
     <description>Comet4J容器侦听</description> 
     <listener-class>org.comet4j.core.CometAppListener</listener-class> 
</listener> 

<servlet> 
    <description>Comet连接[默认:org.comet4j.core.CometServlet]</description>
    <display-name>CometServlet</display-name> 
    <servlet-name>CometServlet</servlet-name>
    <servlet-class>org.comet4j.core.CometServlet</servlet-class> </servlet>

 <servlet-mapping> 
    <servlet-name>CometServlet</servlet-name>
    <url-pattern>/conn</url-pattern>
 </servlet-mapping>

注: 1和2 的位置反过来就会报错

客户端使用简介

客户端是一个JavaScript文件(comet4j-0.0.2.js),其中最重要的是JS.Connector和JS.Engine两个类。JS.Connector负责与服务器建立并保持连接,而JS.Engine类负责将服务器推送过来的消息转化为开发人员可以处理的消息事件,并分发出去,关于客户端的API请参见:http://doc.comet4j.tk/jsdocs/ 。大多数情况下,我们仅需要使用JS.Engine类就可以完成多数的开发工作。

JS.Engine类是一个静态类,在一个页面中只有一个JS.Engine类的实例。它除了负责把服务器推过来的消息转化为事件分发以外,与服务器的连接与断开也由此类负责。

JS.Engine.start方法

JS.Engine.start(String str)和JS.Engine.stop(String str)分别控制连接和断开动作,start方法需要传入一个字符串参数,用来指定您配置的Comet4J连接地址。比如按前面准备工作的配置了CometServlet的地址为/conn,那么可以这样写: JS.Engine.start('/conn'); 上段代码我们让浏览器与服务器进行连接,当连接成功以后JS.Engine类会发出"start"事件,如何进行事件的处理我们稍后介绍。

JS.Engine.stop方法

我们也能够让连接断开: JS.Engine.stop('主动断开'); 上面代码我们让连接断开,并传入了一个“主动断开”这样一个断开的原因。如果您并不需要对断开的原因进行说明,也可以不传递参数: JS.Engine.stop();

JS.Engine类的事件处理

上面我们介绍了如何使用start和stop方法来建立和断开连接,当成功建立连接已后JS.Engine会发出"start"事件,当断开后会发出“stop”事件,当收到某个通道推送过来的信息时也会发出与通道标识同名的事件。您可以事先在中使用JS.Engine.on方法来注册事件处理函数。例如:

JS.Engine.on('start',function(cId, channelList, engine)
     { 
           alert('连接已建立,连接ID为:' + cId);
    }
); 

JS.Engine.on('stop',function(cause, cId, url, engine)
     {  
          alert('连接已断开,连接ID为:' + cId + ',断开原因:' + cause + ',断开的连接地址:'+ url);
    }
); 

也可以将上段代码写成,下面代码与上段代码完全等效:

 JS.Engine.on(
{
       start : function(cId, channelList, engine)
       {
            alert('连接已建立,连接ID为:' + cId); 
      },     
     stop : function(cause, cId, url, engine)
     { 
            alert('连接已断开,连接ID为:' + cId + ',断开原因:' + cause + ',断开的连接地址:'+ url);
      } 
}); 

接下来,介绍一下如何对服务器推送过来的消息进行处理。在介绍之前,我们假设后台已经注册了一个"hello"的应用通道标识,并且只向客户端推送简单的字符串信息。先看如下代码:

JS.Engine.on('hello',function(text){ alert(text); }); 

这样当服务器端使用"hello"通道标识推送过来的消息就可以由上段代码进行处理,将推送过来的信息弹出。

特别注意:以上代码在事件处理函数中使用了alert仅为说明函数功能,实际使用中,在事件处理函数中切勿使用alert、prompt、confirm等可以中断脚本运行的函数,因为Engine需要实时的保持工作状态。

服务器端使用简介

服务端由一个Jar包组成,其中最重的是CometContext和CometEngine两个类。

Comet Context 类

CometContext是一个单态类,通过其getInstance方法来获得实例,它主要负责框架的一些初始化工作保存着一些参数的配置值,除此之外它还有一个更重要的职责——负责注册应用通道标识。如果您想使用框架来实现自己的应用,那么您必需要为自己的应用分配一个唯一的通道标识,并将此通道标识在WEB容器启动时使用CometContext的registChannel方法进行注册,这样,客户端才可以正确接受此应用所推送的消息。注册一个通道标识非常简单:

CometContext.getInstance().registChannel("hello"); 

这样便注册了一个标识为“hello”的应用通道,而客户也可以通过JS.Engine.on('hello',function(msg){...})的形式来接收并处理来自此通道的消息。

Comet Engine 类

另一个重要的类是CometEngine,它除了负责对连接的处理之外,对于开发人员而言,更加常用的可能是它所提供的sendTo或sendToAll方法来向客户端发送消息:

 String channel = "hello"; String someConnectionId = "1125-6634-888";
 engine.sendToAll(channel , "我来了!"); 
engine.sendTo(channel , engine.getConnection(someConnectionId),“Hi,我是XXX”); 

上面代码使用sendToAll方法向所有客户端在"hello"通道上发送了“我来了!”这样一条消息,然后又使用sendTo在同样的通道上向某一个连接发送了“Hi,我是XXX”消息。 CometEngine另外一个很重要的地方在于,它是框架工作的事件引擎的集散地,它提供了BeforeConnectEvent、BeforeDropEvent、ConnectEvent、DropEvent、MessageEvent等事件。通过对这些事件的处理来实现具体的功能:

class JoinListener extends ConnectListener { 
    @Override 
     public boolean handleEvent(ConnectEvent anEvent) {                              
                CometConnection conn = anEvent.getConn();       
                  CometContext.getInstance().getEngine().sendTo("hello",   conn.getId(),"欢迎上线");
  } 
}


CometEngine engine = CometContext.getInstance().getEngine(); 
engine.addConnectListener(new JoinListener());

上面先定义了一个JoinListener并实现了父类ConnectListener的handleEvent抽像方法,然后使用engine.addConnectListener来注册这个事件侦听。这样,在有客户与服务器成功建立连接已后,就可以向此客户端推送一条欢迎信息。
在线Demo体验

http://www.comet4j.org:8080/comet4j/


comet4j.js 源码

有兴趣可以研究下,不是很多

/*
 * Comet4J JavaScript Client V0.1.0
 * Copyright(c) 2011, jinghai.xiao@gamil.com.
 * http://code.google.com/p/comet4j/
 * This code is licensed under BSD license. Use it as you wish, 
 * but keep this copyright intact.
 */
var JS = {
  version: '0.0.2'
};
JS.Runtime = (function () {
  var ua = navigator.userAgent.toLowerCase(),
  check = function (r) {
    return r.test(ua);
  },
  isOpera = check(/opera/),
  isFirefox = check(/firefox/),
  isChrome = check(/chrome/),
  isWebKit = check(/webkit/),
  isSafari = !isChrome && check(/safari/),
  isSafari2 = isSafari && check(/applewebkit\/4/),
  isSafari3 = isSafari && check(/version\/3/),
  isSafari4 = isSafari && check(/version\/4/),
  isIE = !isOpera && check(/msie/),
  isIE7 = isIE && check(/msie 7/),
  isIE8 = isIE && check(/msie 8/),
  isIE6 = isIE && !isIE7 && !isIE8,
  isGecko = !isWebKit && check(/gecko/),
  isGecko2 = isGecko && check(/rv:1\.8/),
  isGecko3 = isGecko && check(/rv:1\.9/),
  isWindows = check(/windows|win32/),
  isMac = check(/macintosh|mac os x/),
  isAir = check(/adobeair/),
  isLinux = check(/linux/);
  return {
    isOpera: isOpera,
    isFirefox: isFirefox,
    isChrome: isChrome,
    isWebKit: isWebKit,
    isSafari: isSafari,
    isSafari2: isSafari2,
    isSafari3: isSafari3,
    isSafari4: isSafari4,
    isIE: isIE,
    isIE7: isIE7,
    isIE8: isIE8,
    isIE6: isIE6,
    isGecko: isGecko,
    isGecko2: isGecko2,
    isGecko3: isGecko3,
    isWindows: isWindows,
    isMac: isMac,
    isAir: isAir,
    isLinux: isLinux
  };
}());
JS.isOpera = JS.Runtime.isOpera;
JS.isFirefox = JS.Runtime.isFirefox;
JS.isChrome = JS.Runtime.isChrome;
JS.isWebKit = JS.Runtime.isWebKit;
JS.isSafari = JS.Runtime.isSafari;
JS.isSafari2 = JS.Runtime.isSafari2;
JS.isSafari3 = JS.Runtime.isSafari3;
JS.isSafari4 = JS.Runtime.isSafari4;
JS.isIE = JS.Runtime.isIE;
JS.isIE7 = JS.Runtime.isIE7;
JS.isIE8 = JS.Runtime.isIE8;
JS.isIE6 = JS.Runtime.isIE6;
JS.isGecko = JS.Runtime.isGecko;
JS.isGecko2 = JS.Runtime.isGecko2;
JS.isGecko3 = JS.Runtime.isGecko3;
JS.isWindows = JS.Runtime.isWindows;
JS.isMac = JS.Runtime.isMac;
JS.isAir = JS.Runtime.isAir;
JS.isLinux = JS.Runtime.isLinux;
JS.Syntax = {
  nameSpace: function () {
    if (arguments.length) {
      var o,
      d,
      v;
      for (var i = 0, len = arguments.length; i < len; i++) {
        v = arguments[i];
        if (!v) {
          continue;
        }
        d = v.split('.');
        for (var j = 0, len = d.length; j < len; j++) {
          if (!d[j]) {
            continue;
          }
          o = window[d[j]] = window[d[j]] || {
          };
        }
      }
    }
    return o;
  },
  apply: function (o, c, defaults) {
    if (defaults) {
      JS.Syntax.apply(o, defaults);
    }
    if (o && c && typeof c == 'object') {
      for (var p in c) {
        o[p] = c[p];
      }
    }
    return o;
  },
  override: function (origclass, overrides) {
    if (overrides) {
      var p = origclass.prototype;
      JS.Syntax.apply(p, overrides);
      if (JS.Runtime.isIE && overrides.hasOwnProperty('toString')) {
        p.toString = overrides.toString;
      }
    }
  },
  extend: function () {
    var io = function (o) {
      for (var m in o) {
        this[m] = o[m];
      }
    };
    var oc = Object.prototype.constructor;
    return function (sb, sp, overrides) {
      if (JS.Syntax.isObject(sp)) {
        overrides = sp;
        sp = sb;
        sb = overrides.constructor != oc ? overrides.constructor : function () {
          sp.apply(this, arguments);
        };
      }
      var F = function () {
      },
      sbp,
      spp = sp.prototype;
      F.prototype = spp;
      sbp = sb.prototype = new F();
      sbp.constructor = sb;
      sb.superclass = spp;
      if (spp.constructor == oc) {
        spp.constructor = sp;
      }
      sb.override = function (o) {
        JS.Syntax.override(sb, o);
      };
      sbp.superclass = sbp.supr = (function () {
        return spp;
      });
      sbp.override = io;
      JS.Syntax.override(sb, overrides);
      sb.extend = function (o) {
        return JS.Syntax.extend(sb, o);
      };
      return sb;
    };
  }(),
  callBack: function (fn, scope, arg) {
    if (JS.isFunction(fn)) {
      return fn.apply(scope || window, arg || [
      ]);
    }
  },
  isEmpty: function (v, allowBlank) {
    return v === null || v === undefined || ((Ext.isArray(v) && !v.length)) || (!allowBlank ? v === '' : false);
  },
  isArray: function (v) {
    return Object.prototype.toString.apply(v) === '[object Array]';
  },
  isDate: function (v) {
    return Object.prototype.toString.apply(v) === '[object Date]';
  },
  isObject: function (v) {
    return !!v && Object.prototype.toString.call(v) === '[object Object]';
  },
  isPrimitive: function (v) {
    return Ext.isString(v) || Ext.isNumber(v) || Ext.isBoolean(v);
  },
  isFunction: function (v) {
    return Object.prototype.toString.apply(v) === '[object Function]';
  },
  isNumber: function (v) {
    return typeof v === 'number' && isFinite(v);
  },
  isString: function (v) {
    return typeof v === 'string';
  },
  isBoolean: function (v) {
    return typeof v === 'boolean';
  },
  isElement: function (v) {
    return !!v && v.tagName;
  },
  isDefined: function (v) {
    return typeof v !== 'undefined';
  },
  toArray: function () {
    return JS.isIE ? function (a, i, j, res) {
      res = [
      ];
      for (var x = 0, len = a.length; x < len; x++) {
        res.push(a[x]);
      }
      return res.slice(i || 0, j || res.length);
    }
     : function (a, i, j) {
      return Array.prototype.slice.call(a, i || 0, j || a.length);
    };
  }()
};
JS.ns = JS.Syntax.nameSpace;
JS.apply = JS.Syntax.apply;
JS.override = JS.Syntax.override;
JS.extend = JS.Syntax.extend;
JS.callBack = JS.Syntax.callBack;
JS.isEmpty = JS.Syntax.isEmpty;
JS.isArray = JS.Syntax.isArray;
JS.isDate = JS.Syntax.isDate;
JS.isObject = JS.Syntax.isObject;
JS.isPrimitive = JS.Syntax.isPrimitive;
JS.isFunction = JS.Syntax.isFunction;
JS.isNumber = JS.Syntax.isNumber;
JS.isString = JS.Syntax.isString;
JS.isBoolean = JS.Syntax.isBoolean;
JS.isElement = JS.Syntax.isElement;
JS.isDefined = JS.Syntax.isDefined;
JS.toArray = JS.Syntax.toArray;
JS.DomEvent = {
  on: function (el, name, fun, scope) {
    if (el.addEventListener) {
      el.addEventListener(name, function () {
        JS.callBack(fun, scope, arguments);
      }, false);
    } else {
      el.attachEvent('on' + name, function () {
        JS.callBack(fun, scope, arguments);
      });
    }
  },
  un: function (el, name, fun, scope) {
    if (el.removeEventListener) {
      el.removeEventListener(name, fun, false);
    } else {
      el.detachEvent('on' + name, fun);
    }
  },
  stop: function (e) {
    e.returnValue = false;
    if (e.preventDefault) {
      e.preventDefault();
    }
    JS.DomEvent.stopPropagation(e);
  },
  stopPropagation: function (e) {
    e.cancelBubble = true;
    if (e.stopPropagation) {
      e.stopPropagation();
    }
  }
};
JS.on = JS.DomEvent.on;
JS.un = JS.DomEvent.un;
JS.DelayedTask = function (fn, scope, args) {
  var me = this,
  id,
  call = function () {
    clearInterval(id);
    id = null;
    fn.apply(scope, args || [
    ]);
  };
  me.delay = function (delay, newFn, newScope, newArgs) {
    me.cancel();
    fn = newFn || fn;
    scope = newScope || scope;
    args = newArgs || args;
    id = setInterval(call, delay);
  };
  me.cancel = function () {
    if (id) {
      clearInterval(id);
      id = null;
    }
  };
};
JS.ns('JS.Observable');
JS.Observable = function (o) {
  JS.apply(this, o || JS.toArray(arguments) [0]);
  if (this.events) {
    this.addEvents(this.events);
  }
  if (this.listeners) {
    this.on(this.listeners);
    delete this.listeners;
  }
};
JS.Observable.prototype = {
  on: function (eventName, fn, scope, o) {
    if (JS.isString(eventName)) {
      this.addListener(eventName, fn, scope, o);
    } else if (JS.isObject(eventName)) {
      this.addListeners(eventName, scope, o);
    }
  },
  fireEvent: function () {
    var arg = JS.toArray(arguments),
    eventName = arg[0].toLowerCase(),
    e = this.events[eventName];
    if (e && !JS.isBoolean(e)) {
      return e.fire.apply(e, arg.slice(1));
    }
  },
  addEvent: function (eventName) {
    if (!JS.isObject(this.events)) {
      this.events = {
      };
    }
    if (this.events[eventName]) {
      return;
    }
    if (JS.isString(eventName)) {
      this.events[eventName.toLowerCase()] = true;
    } else if (eventName instanceof JS.Event) {
      this.events[eventName.name.toLowerCase()] = eventName;
    }
  },
  addEvents: function (arr) {
    if (JS.isArray(arr)) {
      for (var i = 0, len = arr.length; i < len; i++) {
        this.addEvent(arr[i]);
      }
    }
  },
  addListener: function (eventName, fn, scope, o) {
    eventName = eventName.toLowerCase();
    var e = this.events[eventName];
    if (e) {
      if (JS.isBoolean(e)) {
        e = this.events[eventName] = new JS.Event(eventName, this);
      }
      e.addListener(fn, scope, o);
    }
  },
  addListeners: function (obj, scope, o) {
    if (JS.isObject(obj)) {
      for (var p in obj) {
        this.addListener(p, obj[p], scope, o);
      }
    }
  },
  removeListener: function (eventName, fn, scope) {
    eventName = eventName.toLowerCase();
    var e = this.events[eventName];
    if (e && !JS.isBoolean(e)) {
      e.removeListener(fn, scope);
    }
  },
  clearListeners: function () {
    var events = this.events,
    e;
    for (var p in events) {
      e = events[p];
      if (!JS.isBoolean(e)) {
        e.clearListeners();
      }
    }
  },
  clearEvents: function () {
    var events = this.events;
    this.clearListeners();
    for (var p in events) {
      this.removeEvent(p);
    }
  },
  removeEvent: function (eventName) {
    var events = this.events,
    e;
    if (events[eventName]) {
      e = events[eventName];
      if (!JS.isBoolean(e)) {
        e.clearListeners();
      }
      delete events[eventName];
    }
  },
  removeEvents: function (eventName) {
    if (JS.isString(eventName)) {
      this.removeEvent(eventName);
    } else if (JS.isArray(eventName) && eventName.length > 0) {
      for (var i = 0, len = eventName.length; i < len; i++) {
        this.removeEvent(eventName[i]);
      }
    }
  },
  hasEvent: function (eventName) {
    return this.events[eventName] ? true : false;
  },
  hasListener: function (eventName, fn, scope) {
    var events = this.events,
    e = events[eventName];
    if (!JS.isBoolean(e)) {
      return e.hasListener(fn, scope);
    }
    return false;
  },
  suspendEvents: function () {
  },
  resumeEvents: function () {
  }
};
JS.Event = function (name, caller) {
  this.name = name.toLowerCase();
  this.caller = caller;
  this.listeners = [
  ];
};
JS.Event.prototype = {
  fire: function () {
    var
    listeners = this.listeners,
    i = listeners.length - 1;
    for (; i > - 1; i--) {
      if (listeners[i].execute.apply(listeners[i], arguments) === false) {
        return false;
      }
    }
    return true;
  },
  addListener: function (fn, scope, o) {
    scope = scope || this.caller;
    if (this.hasListener(fn, scope) == - 1) {
      this.listeners.push(new JS.Listener(fn, scope, o));
    }
  },
  removeListener: function (fn, scope) {
    var index = this.hasListener(fn, scope);
    alert(index);
    if (index != - 1) {
      this.listeners.splice(index, 1);
    }
  },
  hasListener: function (fn, scope) {
    var i = 0,
    listeners = this.listeners,
    len = listeners.length;
    for (; i < len; i++) {
      if (listeners[i].equal(fn, scope)) {
        return i;
      }
    }
    return - 1;
  },
  clearListeners: function () {
    var i = 0,
    listeners = this.listeners,
    len = listeners.length;
    for (; i < len; i++) {
      listeners[i].clear();
    }
    this.listeners.splice(0);
  }
};
JS.Listener = function (fn, scope, o) {
  this.handler = fn;
  this.scope = scope;
  this.o = o;
};
JS.Listener.prototype = {
  execute: function () {
    return JS.callBack(this.handler, this.scope, arguments);
  },
  equal: function (fn, scope) {
    return this.handler === fn ? true : false;
  },
  clear: function () {
    delete this.handler;
    delete this.scope;
    delete this.o;
  }
};
JS.ns('JS.HTTPStatus', 'JS.XMLHttpRequest');
JS.HTTPStatus = {
  '100': 'Continue',
  '101': 'Switching Protocols',
  '200': 'OK',
  '201': 'Created',
  '202': 'Accepted',
  '203': 'Non-Authoritative Information',
  '204': 'No Content',
  '205': 'Reset Content',
  '206': 'Partial Content',
  '300': 'Multiple Choices',
  '301': 'Moved Permanently',
  '302': 'Found',
  '303': 'See Other',
  '304': 'Not Modified',
  '305': 'Use Proxy',
  '306': 'Unused',
  '307': 'Temporary Redirect',
  '400': 'Bad Request',
  '401': 'Unauthorized',
  '402': 'Payment Required',
  '403': 'Forbidden',
  '404': 'Not Found',
  '405': 'Method Not Allowed',
  '406': 'Not Acceptable',
  '407': 'Proxy Authentication Required',
  '408': 'Request Timeout',
  '409': 'Conflict',
  '410': 'Gone',
  '411': 'Length Required',
  '412': 'Precondition Failed',
  '413': 'Request Entity Too Large',
  '414': 'Request-URI Too Long',
  '415': 'Unsupported Media Type',
  '416': 'Requested Range Not Satisfiable',
  '417': 'Expectation Failed',
  '500': 'Internal Server Error',
  '501': 'Not Implemented',
  '502': 'Bad Gateway',
  '503': 'Service Unavailable',
  '504': 'Gateway Timeout',
  '505': 'HTTP Version Not Supported'
};
JS.HTTPStatus.OK = 200;
JS.HTTPStatus.BADREQUEST = 400;
JS.HTTPStatus.FORBIDDEN = 403;
JS.HTTPStatus.NOTFOUND = 404;
JS.HTTPStatus.TIMEOUT = 408;
JS.HTTPStatus.SERVERERROR = 500;
JS.XMLHttpRequest = JS.extend(JS.Observable, {
  enableCache: false,
  timeout: 0,
  isAbort: false,
  specialXHR: '',
  _xhr: null,
  readyState: 0,
  status: 0,
  statusText: '',
  responseText: '',
  responseXML: null,
  constructor: function () {
    var self = this;
    this.addEvents(['readyStateChange',
    'timeout',
    'abort',
    'error',
    'load',
    'progress']);
    JS.XMLHttpRequest.superclass.constructor.apply(this, arguments);
    this._xhr = this.createXmlHttpRequestObject();
    this._xhr.onreadystatechange = function () {
      self.doReadyStateChange();
    };
  },
  timeoutTask: null,
  delayTimeout: function () {
    if (this.timeout) {
      if (!this.timeoutTask) {
        this.timeoutTask = new JS.DelayedTask(function () {
          if (this._xhr.readyState != 4) {
            this.fireEvent('timeout', this, this._xhr);
          } else {
            this.cancelTimeout();
          }
        }, this);
      }
      this.timeoutTask.delay(this.timeout);
    }
  },
  cancelTimeout: function () {
    if (this.timeoutTask) {
      this.timeoutTask.cancel();
    }
  },
  createXmlHttpRequestObject: function () {
    var activeX = [
      'Msxml2.XMLHTTP.6.0',
      'Msxml2.XMLHTTP.5.0',
      'Msxml2.XMLHTTP.4.0',
      'Msxml2.XMLHTTP.3.0',
      'Msxml2.XMLHTTP',
      'Microsoft.XMLHTTP'
    ],
    xhr,
    specialXHR = this.specialXHR;
    if (specialXHR) {
      if (JS.isString(specialXHR)) {
        return new ActiveXObject(specialXHR);
      } else {
        return specialXHR;
      }
    }
    try {
      xhr = new XMLHttpRequest();
    } catch (e) {
      for (var i = 0; i < activeX.length; ++i) {
        try {
          xhr = new ActiveXObject(activeX[i]);
          break;
        } catch (e) {
        }
      }
    } finally {
      return xhr;
    }
  },
  doReadyStateChange: function () {
    this.delayTimeout();
    var xhr = this._xhr;
    try {
      this.readyState = xhr.readyState;
    } catch (e) {
      this.readyState = 0;
    }
    try {
      this.status = xhr.status;
    } catch (e) {
      this.status = 0;
    }
    try {
      this.statusText = xhr.statusText;
    } catch (e) {
      this.statusText = '';
    }
    try {
      this.responseText = xhr.responseText;
    } catch (e) {
      this.responseText = '';
    }
    try {
      this.responseXML = xhr.responseXML;
    } catch (e) {
      this.responseXML = null;
    }
    this.fireEvent('readyStateChange', this.readyState, this.status, this, xhr);
    if (this.readyState == 3 && (this.status >= 200 && this.status < 300)) {
      this.fireEvent('progress', this, xhr);
    }
    if (this.readyState == 4) {
      this.cancelTimeout();
      var status = this.status;
      if (status == 0) {
        this.fireEvent('error', this, xhr);
      } else if (status >= 200 && status < 300) {
        this.fireEvent('load', this, xhr);
      } else if (status >= 400 && status != 408) {
        this.fireEvent('error', this, xhr);
      } else if (status == 408) {
        this.fireEvent('timeout', this, xhr);
      }
    }
    this.onreadystatechange();
  },
  onreadystatechange: function () {
  },
  open: function (method, url, async, username, password) {
    if (!url) {
      return;
    }
    if (!this.enableCache) {
      if (url.indexOf('?') != - 1) {
        url += '&ram=' + Math.random();
      } else {
        url += '?ram=' + Math.random();
      }
    }
    this._xhr.open(method, url, async, username, password);
  },
  send: function (content) {
    this.delayTimeout();
    this.isAbort = false;
    this._xhr.send(content);
  },
  abort: function () {
    this.isAbort = true;
    this.cancelTimeout();
    this._xhr.abort();
    if (JS.isIE) {
      var self = this;
      self._xhr.onreadystatechange = function () {
        self.doReadyStateChange();
      };
    }
    this.fireEvent('abort', this, this._xhr);
  },
  setRequestHeader: function (header, value) {
    this._xhr.setRequestHeader(header, value);
  },
  getResponseHeader: function (header) {
    return this._xhr.getResponseHeader(header);
  },
  getAllResponseHeaders: function () {
    return this._xhr.getAllResponseHeaders();
  },
  setTimeout: function (t) {
    this.timeout = t;
  }
});
JS.ns('JS.AJAX');
JS.AJAX = (function () {
  var xhr = new JS.XMLHttpRequest();
  return {
    dataFormatError: '服务器返回的数据格式有误',
    urlError: '未指定url',
    post: function (url, param, callback, scope, asyn) {
      if (typeof url !== 'string') {
        throw new Error(this.urlError);
      }
      var asynchronous = true;
      if (asyn === false) {
        asynchronous = false;
      }
      xhr.onreadystatechange = function () {
        if (xhr.readyState == 4 && asynchronous) {
          JS.callBack(callback, scope, [
            xhr
          ]);
        }
      };
      xhr.open('POST', url, asynchronous);
      xhr.setRequestHeader('Content-Type', 'application/x-www-form-urlencoded;charset=UTF8');
      xhr.send(param || null);
      if (!asynchronous) {
        JS.callBack(callback, scope, [
          xhr
        ]);
      }
    },
    get: function (url, param, callback, scope, asyn) {
      if (typeof url !== 'string') {
        throw new Error(this.urlError);
      }
      var asynchronous = true;
      if (asyn === false) {
        asynchronous = false;
      }
      xhr.onreadystatechange = function () {
        if (xhr.readyState == 4 && asynchronous) {
          JS.callBack(callback, scope, [
            xhr
          ]);
        }
      };
      xhr.open('GET', url, asynchronous);
      xhr.setRequestHeader('Content-Type', 'html/text;charset=UTF8');
      xhr.send(param || null);
      if (!asynchronous) {
        JS.callBack(callback, scope, [
          xhr
        ]);
      }
    },
    getText: function (url, jsonData, callback, scope, asyn) {
      this.get(url, jsonData, function (xhr) {
        if (scope) {
          callback.call(scope, xhr.responseText);
        } else {
          callback(xhr.responseText);
        }
      }, this, asyn);
    },
    getJson: function (url, jsonData, callback, scope, asyn) {
      this.get(url, jsonData, function (xhr) {
        var json = null;
        try {
          json = eval('(' + xhr.responseText + ')');
        } catch (e) {
          throw new Error(this.dataFormatError);
        }
        JS.callBack(callback, scope, [
          json
        ]);
      }, this, asyn);
    }
  };
}) ();
JS.ns('JS.Connector');
JS.Connector = JS.extend(JS.Observable, {
  version: '0.0.2',
  SYSCHANNEL: 'c4j',
  LLOOPSTYLE: 'lpool',
  STREAMSTYLE: 'stream',
  CMDTAG: 'cmd',
  url: '',
  param: '',
  revivalDelay: 100,
  cId: '',
  channels: [
  ],
  workStyle: '',
  emptyUrlError: 'URL为空',
  runningError: '连接正在运行',
  dataFormatError: '数据格式有误',
  running: false,
  _xhr: null,
  lastReceiveMessage: '',
  constructor: function () {
    JS.Connector.superclass.constructor.apply(this, arguments);
    this.addEvents(['beforeConnect',
    'connect',
    'beforeStop',
    'stop',
    'message',
    'revival']);
    if (JS.isIE7) {
      this._xhr = new JS.XMLHttpRequest({
        specialXHR: 'Msxml2.XMLHTTP.6.0'
      });
    } else {
      this._xhr = new JS.XMLHttpRequest();
    }
    this._xhr.addListener('progress', this.doOnProgress, this);
    this._xhr.addListener('load', this.doOnLoad, this);
    this._xhr.addListener('error', this.doOnError, this);
    this._xhr.addListener('timeout', this.revivalConnect, this);
    this.addListener('beforeStop', this.doDrop, this);
    JS.on(window, 'beforeunload', this.doDrop, this);
  },
  doDrop: function (url, cId, conn, xhr) {
    if (!this.running || !this.cId) {
      return;
    }
    try {
      var xhr = new JS.XMLHttpRequest();
      var url = this.url + '?' + this.CMDTAG + '=drop&cid=' + this.cId;
      xhr.open('GET', url, false);
      xhr.send(null);
      xhr = null;
    } catch (e) {
    };
  },
  dispatchServerEvent: function (msg) {
    this.fireEvent('message', msg.channel, msg.data, msg.time, this);
  },
  translateStreamData: function (responseText) {
    var str = responseText;
    if (this.lastReceiveMessage && str) {
      str = str.split(this.lastReceiveMessage);
      str = str.length ? str[str.length - 1] : '';
    }
    this.lastReceiveMessage = responseText;
    return str;
  },
  decodeMessage: function (msg) {
    var json = null;
    if (JS.isString(msg) && msg != '') {
      if (msg.charAt(0) == '<') {
        msg = msg.substring(1, msg.length);
      }
      if (msg.charAt(msg.length - 1) == '>') {
        msg = msg.substring(0, msg.length - 1);
      }
      msg = decodeURIComponent(msg);
      try {
        json = eval('(' + msg + ')');
      } catch (e) {
        this.stop('JSON转换异常');
        try {
          console.log('JSON转换异常:' + msg);
        } catch (e) {
        };
      }
    }
    return json;
  },
  doOnProgress: function (xhr) {
    if (this.workStyle === this.STREAMSTYLE) {
      var str = this.translateStreamData(xhr.responseText);
      var msglist = str.split('>');
      if (msglist.length > 0) {
        for (var i = 0, len = msglist.length; i < len; i++) {
          var json = this.decodeMessage(msglist[i]);
          if (json) {
            this.dispatchServerEvent(json);
          }
        }
      }
    }
  },
  doOnError: function (xhr) {
    this.stop('服务器异常');
  },
  doOnLoad: function (xhr) {
    if (this.workStyle === this.LLOOPSTYLE) {
      var json = this.decodeMessage(xhr.responseText);
      if (json) {
        this.dispatchServerEvent(json);
      }
    }
    this.revivalConnect();
  },
  startConnect: function () {
    if (this.running) {
      var url = this.url + '?' + this.CMDTAG + '=conn&cv=' + this.version + this.param;
      JS.AJAX.get(url, '', function (xhr) {
        var msg = this.decodeMessage(xhr.responseText);
        if (!msg) {
          this.stop('连接失败');
          return;
        }
        var data = msg.data;
        this.cId = data.cId;
        this.channels = data.channels;
        this.workStyle = data.ws;
        this._xhr.setTimeout(data.timeout * 2);
        this.fireEvent('connect', data.cId, data.channels, data.ws, data.timeout, this);
        this.revivalConnect();
      }, this);
    }
  },
  revivalConnect: function () {
    var self = this;
    if (this.running) {
      setTimeout(revival, this.revivalDelay);
    }
    function revival() {
      var xhr = self._xhr;
      var url = self.url + '?' + self.CMDTAG + '=revival&cid=' + self.cId + self.param;
      xhr.open('GET', url, true);
      xhr.send(null);
      self.fireEvent('revival', self.url, self.cId, self);
    }
  },
  start: function (url, param) {
    var self = this;
    setTimeout(function () {
      if (!self.url && !url) {
        throw new Error(self.emptyUrlError);
      }
      if (self.running) {
        return;
      }
      if (url) {
        self.url = url;
      }
      if (param && JS.isString(param)) {
        if (param.charAt(0) != '&') {
          param = '&' + param;
        }
        self.param = param;
      }
      if (self.fireEvent('beforeConnect', self.url, self) === false) {
        return;
      }
      self.running = true;
      self.startConnect();
    }, 1000);
  },
  stop: function (cause) {
    if (!this.running) {
      return;
    }
    if (this.fireEvent('beforeStop', cause, this.cId, this.url, this) === false) {
      return;
    }
    this.running = false;
    var cId = this.cId;
    this.cId = '';
    this.param = '';
    this.adml = [
    ];
    this.workStyle = '';
    try {
      this._xhr.abort();
    } catch (e) {
    };
    this.fireEvent('stop', cause, cId, this.url, this);
  },
  getId: function () {
    return this.cId;
  }
});
JS.ns('JS.Engine');
JS.Engine = (function () {
  var Engine = JS.extend(JS.Observable, {
    lStore: [
    ],
    running: false,
    connector: null,
    constructor: function () {
      this.addEvents(['start',
      'stop']);
      Engine.superclass.constructor.apply(this, arguments);
      this.connector = new JS.Connector();
      this.initEvent();
    },
    addListener: function (eventName, fn, scope, o) {
      if (this.running) {
        Engine.superclass.addListener.apply(this, arguments);
      } else {
        this.lStore.push({
          eventName: eventName,
          fn: fn,
          scope: scope,
          o: o
        });
      }
    },
    initEvent: function () {
      var self = this;
      this.connector.on({
        connect: function (cId, aml, conn) {
          self.running = true;
          self.addEvents(aml);
          for (var i = 0, len = self.lStore.length; i < len; i++) {
            var e = self.lStore[i];
            self.addListener(e.eventName, e.fn, e.scope);
          }
          self.fireEvent('start', cId, aml, self);
        },
        stop: function (cause, cId, url, conn) {
          self.running = false;
          self.fireEvent('stop', cause, cId, url, self);
          self.clearListeners();
        },
        message: function (amk, data, time) {
          self.fireEvent(amk, data, time, self);
        }
      });
    },
    start: function (url) {
      if (this.running) {
        return;
      }
      this.connector.start(url);
    },
    stop: function (cause) {
      if (!this.running) {
        return;
      }
      this.connector.stop(cause);
    },
    getConnector: function () {
      return this.connector;
    },
    getId: function () {
      return this.connector.cId;
    }
  });
  return new Engine();
}());

连接断开过程

  • Client Start
    Client run start()
    Client:start
    server: Dying--> Connect-->Revival

  • Time interval death reactivating
    After time interval
    Server:Dying-->Revival
    注:可以当心跳使用,不需要单独的创建心跳机制

  • Client Stop
    Client run Stop()
    Client:stop
    Server:drop

  • Network Error
    After time interval
    Client:stop
    Server:Dying-->drop

测试代码

监听方法一部分外部类继承然后添加,一部分直接使用匿名类,外部的没有贴代码,基本一样

package com.jony.hcsdemo2;

import org.comet4j.core.CometContext;
import org.comet4j.core.CometEngine;
import org.comet4j.core.event.DyingEvent;
import org.comet4j.core.event.RemovedEvent;
import org.comet4j.core.event.RevivalEvent;
import org.comet4j.core.listener.DyingListener;
import org.comet4j.core.listener.RemovedListener;
import org.comet4j.core.listener.RevivalListener;

import javax.servlet.ServletContextEvent;
import javax.servlet.ServletContextListener;

/**
 * Created by jony on 11/24/17.
 */
public class ConnectListener implements ServletContextListener{


//   控制指令频道
    private static final String CONTROL_CHANNEL = "control";

    /**
     * 初始化上下文
     */
    @Override
    public void contextInitialized(ServletContextEvent servletContextEvent) {

        //Comet4J上下文,负责初始化配置、引擎对象、连接器对象、消息缓存等。
        CometContext cc = CometContext.getInstance();

        // 注册频道,即标识哪些字段可用当成频道,用来作为向前台传送数据的“通道”
        cc.registChannel(CONTROL_CHANNEL);


        CometEngine engine = cc.getEngine();
        engine.addConnectListener(new JoinListener());
        engine.addDropListener(new MyDropListener());

        engine.addRevivalListener(new RevivalListener() {
            @Override
            public boolean handleEvent(RevivalEvent revivalEvent) {

                System.out.println("Client Revival: "+revivalEvent.getConn().getId());
                return true;
            }
        });

        engine.addRemovedListener(new RemovedListener() {
            @Override
            public boolean handleEvent(RemovedEvent removedEvent) {

                System.out.println("Client Removed: "+removedEvent.getConn().getId());
                return true;
            }
        });

        engine.addDyingListener(new DyingListener() {
            @Override
            public boolean handleEvent(DyingEvent dyingEvent) {

                System.out.println("Client Dying: "+dyingEvent.getConn().getId());
                return false;
            }
        });
    }


    /**
     * 销毁
     */
    @Override
    public void contextDestroyed(ServletContextEvent servletContextEvent) {

    }
}

$(document).ready(function () {


    setTimeout(cometProcess);
});

function startConn() {
    JS.Engine.start("conn");

    // setTimeout(cometProcess);
    // cometProcess();
}

function dropConn() {
    JS.Engine.stop("client request drop connect");
}

function cometProcess() {
    // JS.Engine.on('start',function(cId, channelList, engine){
    //     alert('连接已建立,连接ID为:' + cId);
    // });
    // JS.Engine.on('stop',function(cause, cId, url, engine){
    //     alert('连接已断开,连接ID为:' + cId + ',断开原因:' + cause + ',断开的连接地址:'+ url);
    // });
    
    JS.Engine.on({
        start : function (cId, channelList, engine) {
            // alert('连接已建立,连接ID为:' + cId);
            $("#status").text('连接已建立,连接ID为:' + cId);
        },
        stop : function (cause, cId, url, engine) {
            // alert('连接已断开,连接ID为:' + cId + ',断开原因:' + cause + ',断开的连接地址:'+ url);
            $("#status").text('连接已断开,连接ID为:' + cId + ',断开原因:' + cause + ',断开的连接地址:'+ url);
        },
        control: function (str) {
            // alert(str);
            $("#msgShow").append(str+"<br>");
        }
        
    });
}
web.xml 配置,服务器当然也要配置,更换为NOI连接,上面
<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_3_1.xsd"
         version="3.1">
    
    <!--配置监听器-->
    <listener>
        <description>Comet Listener</description>
        <listener-class>org.comet4j.core.CometAppListener</listener-class>
    </listener>
    
    <listener>
        <description>Connect Listener</description>
        <listener-class>com.jony.hcsdemo2.ConnectListener</listener-class>
    </listener>
    
    
    
    <!--配置访问入口-->
    <servlet>
        <servlet-name>CometServlet</servlet-name>
        <servlet-class>org.comet4j.core.CometServlet</servlet-class>
    </servlet>
    
    <servlet-mapping>
        <servlet-name>CometServlet</servlet-name>
        <url-pattern>/conn</url-pattern>
    </servlet-mapping>
    
    
    
    
    <welcome-file-list>
        <welcome-file>index.jsp</welcome-file>
    </welcome-file-list>
    
</web-app>

comet 4j lib 里面最常用的一些,算是可以应付基本使用,Google上面的API 文档链接也打不开,大家只能直接打开lib 研究了,当然有找到资源的还请分享下

image.png
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,539评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,911评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,337评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,723评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,795评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,762评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,742评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,508评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,954评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,247评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,404评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,104评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,736评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,352评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,557评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,371评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,292评论 2 352

推荐阅读更多精彩内容