comet

有一种技术是后服务端动推送信息给客户端的,这种技术的名字叫comet,我惊呆了,因为完全没听过,赶紧上网搜集资料,耗了一个晚上写了个简单的例子,实现主动向客户端发送信息。说是说主动,其实还是要客户端先献出它的“第一次”,即只要它有先请求你一下,以后你们熟了,你想主动约它就约它!

  关于comet技术介绍及其实现原理,可以参考网站 http://www.ibm.com/developerworks/cn/web/wa-lo-comet/ 的介绍。

  简单来说,就是客户端发送请求到服务端,服务器端会阻塞请求直到有数据传递或超时才返回,之后客户端 JavaScript 响应处理函数会在处理完服务器返回的信息后,再次发出请求,重新建立连接。当客户端处理接收的数据、重新建立连接时,服务器端可能有新的数据到达;这些信息会被服务器端保存直到客户端重新建立连接,客户端会一次把当前服务器端所有的信息取回。

【工作环境】

1、myeclipse2013

2、tomcat 6.0

3、jdk 7

4、火狐浏览器

说明: 

测试成功的浏览器有:(1)火狐浏览器  (2)IE10、IE9、IE8  (3)360极速浏览器极速模式

测试失败的浏览器有:(1)IE10兼容模式、IE7

【准备工作】

1、下载comet4j.js  :http://files.cnblogs.com/xiaoMzjm/comet4j.js.rar

2、下载comet4j-tomcat6.jar  :http://files.cnblogs.com/xiaoMzjm/comet4j-tomcat6.jar.rar

3、到tomcat目录下——conf——server.xml 下,把

    <Connector 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"
/>  

说明: 

其实那个js文件和jar官网是https://code.google.com/p/comet4j/ 的,但它是谷歌,这里是天朝呐,所以贴了两个我文件夹里面的包的地址上来。    

comet4j-tomcat6.jar 还有另一个版本是 comet4j-tomcat7.jar , 自己选择合适的版本去下载。6以下的tomcat肯定不行就对了。

comet4j.js 的官方使用文档: http://doc.comet4j.tk/jsdocs/

comet4j-tomcat6.jar 的官方使用文档: http://doc.comet4j.tk/apidocs/

【新建项目过程】

(1)新建服务端的类TestComet , 实现 ServletContextListener 接口

(2)在web.xml 里面应该配置 拦截器:

复制代码

    <listener>
        <listener-class>org.comet4j.core.CometAppListener</listener-class>
    </listener>
    <listener>
        <description>HelloWorld</description>
        <listener-class>com.zjm.www.test.TestComet</listener-class>
    </listener>
    <servlet>
        <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>

复制代码

注:其中的要配置的有两个地方

一个是comet4j-tomcat6.jar下的一个servlet:org.comet4j.core.CometServlet , 客户端访问的入口

另一个是comet4j-tomcat6.jar下的监听器:org.comet4j.core.CometAppListener , 监听我们自己的类。

【具体代码(说明都写在注释里面)】

1、web.xml

复制代码

 <?xml version="1.0" encoding="UTF-8"?>
  <web-app version="2.5" 
      xmlns="http://java.sun.com/xml/ns/javaee" 
      xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" 
      xsi:schemaLocation="http://java.sun.com/xml/ns/javaee 
      http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd">
      <listener>
          <listener-class>org.comet4j.core.CometAppListener</listener-class>
     </listener>
     <listener>
         <description>HelloWorld</description>
         <listener-class>com.zjm.www.test.TestComet</listener-class>
     </listener>
     <servlet>
         <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>


   <welcome-file-list>
     <welcome-file>index.jsp</welcome-file>
   </welcome-file-list>
 </web-app>

复制代码

2、java类TestComet

里面附上了不少的注释,如果想仔细研究建议看上面的赋予的API文档链接。

复制代码

 package com.zjm.www.test;

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

 import org.comet4j.core.CometContext;
 import org.comet4j.core.CometEngine;

 /**
  * 描述:服务端主动推送消息到客户端  简单例子
  * @author zjm
  * @time 2014/8/7
  */
 public class TestComet implements ServletContextListener {

         // 频道1
         private static final String CHANNEL1 = "result1";
         // 频道2
         private static final String CHANNEL2 = "result2";

         // 通过频道1推送给前台的变量1
         private static int number1 = 0 ;
         // 通过频道2推送给前台的变量2
         private static int number2 = 100 ;

         /**
          * 初始化上下文
          */
         public void contextInitialized(ServletContextEvent arg0) {

                 // CometContext : Comet4J上下文,负责初始化配置、引擎对象、连接器对象、消息缓存等。
                 CometContext cc = CometContext.getInstance();
                 // 注册频道,即标识哪些字段可用当成频道,用来作为向前台传送数据的“通道”
                 cc.registChannel(CHANNEL1);
                 cc.registChannel(CHANNEL2);

                Thread myThread = new Thread(new SendToClientThread(), "SendToClientThread");
                 // 下面的内部类的方法是个死循环,设置helloAppModule线程为“守护线程”,则当jvm只剩“守护线程”时(主线程结束),该线程也会结束。
                 myThread.setDaemon(true);
                 // 开始线程
                 myThread.start();
         }

         /**
          * 内部类线程类
          */
         class SendToClientThread implements Runnable {
                 public void run() {
                        while (true) {
                                 try {
                                         Thread.sleep(1000);
                                 } catch (Exception ex) {
                                        ex.printStackTrace();
                                 }
                                 // CometEngine : 引擎,负责
管理和维持连接,并能够必要的发送服务
                                 CometEngine engine = CometContext.getInstance().getEngine();
                                 // 参数的意思:通过什么频道(CHANNEL1)发送什么数据(number1++),前台可用可用频道的值(result1)来获取某频道发送的数据
                                 engine.sendToAll(CHANNEL1, number1++);
                                 engine.sendToAll(CHANNEL2, number2++);
                         }
                 }
         }

         public void contextDestroyed(ServletContextEvent arg0) {
         }
 }

复制代码

3、客户端代码

复制代码

<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
  <html xmlns="http://www.w3.org/1999/xhtml">
  <head>
 <meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
  <title>Comet4J Hello World</title>
  <script type="text/javascript" src="js/comet4j.js"></script>
  <script type="text/javascript">
  function init(){

         var number1 = document.getElementById('number1');
         var number2 = document.getElementById('number2');
        // 建立连接,conn 即web.xml中 CometServlet的<url-pattern>
         JS.Engine.start('conn');
         // 监听后台某个频道
         JS.Engine.on(
                { 
                   // 对应服务端 “频道1” 的值 result1
                   result1 : function(num1){
                       number1.innerHTML = num1;
                 },
                 // 对应服务端 “频道2” 的值 result2
                 result2 : function(num2){
                        number2.innerHTML = num2;
                 },
             }
            );
 }
 </script>
 </head>
 <body onload="init()">
         数字1:<span id="number1">...</span><br></br>
         数字2:<span id="number2">...</span>
 </body>
 </html>
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值