服务器主动向客户端推送消息-java之comet4j实现
传统的服务器客户端模型是:客户端发起请求-服务器响应-发送响应结果到客户端。而有些场景是需要在服务端有 相关信息更新时,主动向客户端推送该信息。实现这种模型现已有相关的框架。本博客将介绍comet4j实现web方式的推送!
本博文将简单介绍两个demo,其中一个是时钟,一个是简单数字变化。至于comet4j的介绍,在这里不在累赘( 不然就太啰嗦了!)
打开eclipse,将tomcat7弄进来,新建一个dynamic web project(2.5),放入comet4j-tomcat7.jar,在webContent新建js文件夹,放入comet4j.js,整体的截图如下:
其中,comet4j需要的Jar包与js在这里下载即可:comet4j相关资料下载
除此之外,需要修改tomcat7的server.xml配置:
即:tomcat知道的配置为:
现在需要修改为:
protocol="org.apache.coyote.http11.Http11NioProtocol" redirectPort="8443"/>
新建包:org.steadyjack.comet,在包下新建两个类,TestComet.java和TestCometTimer,均实现ServletContextListener,相关代码如下所示:
TestComet类:
package org.steadyjack.comet;
import javax.servlet.ServletContextEvent;
import javax.servlet.ServletContextListener;
import org.comet4j.core.CometContext;
import org.comet4j.core.CometEngine;
public class TestComet implements ServletContextListener {
// 频道1
private static final String CHANNEL1 = "res1";
// 频道2
private static final String CHANNEL2 = "res2";
// 通过频道1推送给前台的变量1
private static int number1 = 0;
// 通过频道2推送给前台的变量2
private static int number2 = 100;
@Override
public void contextDestroyed(ServletContextEvent arg0) {
}
@Override
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++);
}
}
}
}
package org.steadyjack.comet;
import java.text.SimpleDateFormat;
import java.util.Date;
import javax.servlet.ServletContextEvent;
import javax.servlet.ServletContextListener;
import org.comet4j.core.CometContext;
import org.comet4j.core.CometEngine;
public class TestCometTimer implements ServletContextListener {
// 频道1
private static final String CHANNEL1 = "timer";
@Override
public void contextDestroyed(ServletContextEvent arg0) {
}
@Override
public void contextInitialized(ServletContextEvent arg0) {
// CometContext : Comet4J上下文,负责初始化配置、引擎对象、连接器对象、消息缓存等。
CometContext cc = CometContext.getInstance();
// 注册频道,即标识哪些字段可用当成频道,用来作为向前台传送数据的“通道”
cc.registChannel(CHANNEL1);
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();
//设置日期格式
SimpleDateFormat df = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
// new Date()为获取当前系统时间
String strDate=df.format(new Date());
// 参数的意思:通过什么频道(CHANNEL1)发送什么数据(number1++),前台可用可用频道的值(result1)来获取某频道发送的数据
engine.sendToAll(CHANNEL1, strDate);
}
}
}
}
cometProject
index.jsp
org.comet4j.core.CometAppListener
org.steadyjack.comet.TestComet
org.steadyjack.comet.TestCometTimer
cometServlet
org.comet4j.core.CometServlet
cometServlet
/comet
index.jsp:
<%@ page language="java" contentType="text/html; charset=UTF-8"
pageEncoding="UTF-8"%>
Insert title here
数字1:...
数字2:...
<%@ page language="java" contentType="text/html; charset=UTF-8"
pageEncoding="UTF-8"%>
Insert title here
现在时间是:...
访问indexTime.jsp效果图:
两个图都是不断在变化的!好了,就介绍到这里吧,下文介绍介绍另外一种机制。