- 浏览: 217725 次
- 性别:
- 来自: 天津
文章分类
最新评论
-
flychao88:
<div class="quote_title ...
nio与io的比较 -
李冰冰:
如果NIO单个线程处理业务逻辑,那么如果这个业务逻辑非常耗时, ...
nio与io的比较 -
wj_126mail:
IO是一个连接就创建一个线程来处理;NIO是一个线程在处理所有 ...
nio与io的比较 -
nothingismao:
<div class="quote_title ...
nio与io的比较 -
623deyingxiong:
<div class="quote_title ...
nio与io的比较
在我的认识当中以前一直有一种误区认为:tomcat=web。在我看过了soap协议之后,忽然有了灵感,为什么不能用tomcat来做一个聊天软件的服务器呢?
这个具体的设计如下:
在http协议中嵌入xml(仿照soap)利用xstream把pojo转化成xml然后在服务器、客户端之间传输。如果考虑扩展性,甚至可以把客户端采用c++或者其他语言来编写,当然,其中的解析http\解析xml会比较痛苦,不知道c++有没有类似commons-httpclient\xstream的开源包。总之目前实现的客户端采用的是swt技术。
下面的几个类是一对pojo也就是通讯协议。在客户端、服务器内部使用。需要传输的时候利用xstream转化为xml传输。
协议方面主要有2组:heartbeat\upload。
HeartBeatRequest就是心跳请求,用于请求下在传送给该用户的消息,HeartBeatResponse就是返回类。uploadRequest是发送给其他用户。UploadResponse就是发送后的反馈。
因为http是无状态连接,所以每次请求都需要验证,所以任何XXXRequest都继承自Request类,其中包含验证信息,和发信人地址。protocol的代码就不贴了,太恶心了都是pojo要是想看就自己看代码吧。
下面是一个util类,用作把pojo转化为xml并且嵌入到http协议中并发送到固定url并且接受反馈xml并转化为pojo供client端使用具体代码如下:
MessagePoster::initXStream()是初始化xstream的一个函数,具体从配置文件当中读取。具体配置文件如下:
TimedQueue是该项目的另一个核心类,主要是一个消息队列,采用lru算法维护生命周期。
最后一个核心类就是:SelfCleaner他会定期清理队列中的无用成员,说白了就是长期没有响应的消息。
最后一个类就是服务器类,很短,但很精悍。
写完了。好累。
最后写下怎么用吧:在ChatClient中修改usrId、distId成员来确定发送者和接受者。设成相反的两个数字就可以,当然,客户端要启动2次。验证功能目前是关闭的。password无所谓了。
MessagePoster中的url可能需要修改一下,如果你部署的时候改变路径了。
就这么多了
这个具体的设计如下:
在http协议中嵌入xml(仿照soap)利用xstream把pojo转化成xml然后在服务器、客户端之间传输。如果考虑扩展性,甚至可以把客户端采用c++或者其他语言来编写,当然,其中的解析http\解析xml会比较痛苦,不知道c++有没有类似commons-httpclient\xstream的开源包。总之目前实现的客户端采用的是swt技术。
下面的几个类是一对pojo也就是通讯协议。在客户端、服务器内部使用。需要传输的时候利用xstream转化为xml传输。
协议方面主要有2组:heartbeat\upload。
HeartBeatRequest就是心跳请求,用于请求下在传送给该用户的消息,HeartBeatResponse就是返回类。uploadRequest是发送给其他用户。UploadResponse就是发送后的反馈。
因为http是无状态连接,所以每次请求都需要验证,所以任何XXXRequest都继承自Request类,其中包含验证信息,和发信人地址。protocol的代码就不贴了,太恶心了都是pojo要是想看就自己看代码吧。
下面是一个util类,用作把pojo转化为xml并且嵌入到http协议中并发送到固定url并且接受反馈xml并转化为pojo供client端使用具体代码如下:
public class MessagePoster<REQ_OBJ, RESP_OBJ> { private String url = "http://localhost:8080/HttpChat/Server"; private String propLoc = "com/cxz/httpchat/util/class.properties"; // The XStream instance is thread-safe. public XStream xstream = null; public MessagePoster() { initXStream(); } public void setUrl(String url) { this.url = url; } private void initXStream() { xstream = new XStream(); Properties properties = new Properties(); try { properties.load(MessagePoster.class.getClassLoader() .getResourceAsStream(propLoc)); Set keys = properties.keySet(); for (Object key : keys) { xstream.alias((String) key, Class.forName((String) properties .get(key))); } } catch (IOException e) { // TODO Auto-generated catch block e.printStackTrace(); } catch (ClassNotFoundException e) { // TODO Auto-generated catch block e.printStackTrace(); } } public RESP_OBJ postXML(REQ_OBJ requestObj) throws HttpException, IOException { RESP_OBJ responseObj = null; PostMethod method = null; try { String xml = xstream.toXML(requestObj); HttpClient client = new HttpClient(); method = new PostMethod(url); RequestEntity entity; entity = new StringRequestEntity(xml, "text/xml", "utf-8"); method.setRequestEntity(entity); client.executeMethod(method); responseObj = (RESP_OBJ) xstream.fromXML(method .getResponseBodyAsStream()); } catch(StreamException e){ e.printStackTrace(); } finally { method.releaseConnection(); } return responseObj; } }
MessagePoster::initXStream()是初始化xstream的一个函数,具体从配置文件当中读取。具体配置文件如下:
HeartBeatRequest=com.cxz.httpchat.message.HeartBeatRequest HeartBeatResponse=com.cxz.httpchat.message.HeartBeatResponse UploadRequest=com.cxz.httpchat.message.UploadRequest UploadResponse=com.cxz.httpchat.message.UploadResponse Message=com.cxz.httpchat.message.Message
TimedQueue是该项目的另一个核心类,主要是一个消息队列,采用lru算法维护生命周期。
public class TimedQueue implements Queue<Message> { private long lastAccess; private int cycleSeconds; private static final int UNIT_CONVERSION = 1000; private Queue<Message> queue = new ConcurrentLinkedQueue<Message>(); public List<Message> toList(){ resetLastAccess(); List<Message> messages = new ArrayList<Message>(); while(!queue.isEmpty()){ messages.add(queue.remove()); } return messages; } public boolean isOutOfDate(){ System.out.println(System.currentTimeMillis() - lastAccess); return System.currentTimeMillis() >= lastAccess + cycleSeconds * UNIT_CONVERSION; } public boolean add(Message e) { resetLastAccess(); queue.add(e); return true; } public Message poll() { resetLastAccess(); return queue.poll(); }
最后一个核心类就是:SelfCleaner他会定期清理队列中的无用成员,说白了就是长期没有响应的消息。
package com.cxz.httpchat.util; import java.util.ArrayList; import java.util.List; import java.util.Map; import java.util.Set; import java.util.concurrent.ConcurrentHashMap; import com.cxz.httpchat.message.Message; import com.cxz.httpchat.util.TimedQueue; public class SelfCleaner implements Runnable{ private Map<Integer, TimedQueue> map = new ConcurrentHashMap<Integer, TimedQueue>(); private Thread gcThread; private static SelfCleaner instance; private static final int POLL_SECONDS = 60; private boolean runFlag = true; private static final int UNIT_CONVERSION = 1000; public static synchronized SelfCleaner getInstance() { if (instance == null) { instance = new SelfCleaner(); } return instance; } private SelfCleaner(){ } public void stopGc(){ runFlag = false; gcThread.interrupt(); } public void forceGc() { //Interrupted the sleeping gc Thread gcThread.interrupt(); } public boolean isStopped() { return runFlag; } public void start() { gcThread = new Thread(this, "gcThread"); gcThread.start(); } public void run() { cleanTheHashMap(); while(runFlag){ try { Thread.sleep(POLL_SECONDS * UNIT_CONVERSION); } catch (InterruptedException e) { // This thread has been woken up by a interruption. System.out.println("Hey, man, get up from the bed and gc."); } } } public void remove(Integer id) { map.remove(id); } public List<Message> getMessage(Integer id){ if(map.get(id) == null){ List<Message> list = new ArrayList<Message>(); return list; } else { return map.get(id).toList(); } } public void add(Integer id, Message message) { TimedQueue queue = map.get(id); if (queue == null) {// If the user does not exists in the queue TimedQueue temp = new TimedQueue(); temp.add(message); map.put(id, temp); } else { queue.add(message); } } private void cleanTheHashMap() { Set<Integer> set = map.keySet(); for (Integer id : set) { TimedQueue tq = map.get(id); if (tq.isOutOfDate()) { map.remove(id); } } } }
最后一个类就是服务器类,很短,但很精悍。
package com.cxz.httpchat.server; import java.io.IOException; import java.util.List; import javax.servlet.ServletException; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import com.cxz.httpchat.message.HeartBeatRequest; import com.cxz.httpchat.message.HeartBeatResponse; import com.cxz.httpchat.message.Message; import com.cxz.httpchat.message.UploadRequest; import com.cxz.httpchat.message.UploadResponse; import com.cxz.httpchat.util.MessagePoster; import com.cxz.httpchat.util.SelfCleaner; import com.thoughtworks.xstream.XStream; /** * Servlet implementation class for Servlet: Server * */ public class Server extends javax.servlet.http.HttpServlet implements javax.servlet.Servlet { /* (non-Java-doc) * @see javax.servlet.http.HttpServlet#HttpServlet() */ private XStream xstream = new MessagePoster().xstream; SelfCleaner map = SelfCleaner.getInstance(); public Server() { super(); map.start(); } /* (non-Java-doc) * @see javax.servlet.http.HttpServlet#doGet(HttpServletRequest request, HttpServletResponse response) */ protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { // TODO Auto-generated method stub } /* (non-Java-doc) * @see javax.servlet.http.HttpServlet#doPost(HttpServletRequest request, HttpServletResponse response) */ protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { Object req = xstream.fromXML(request.getInputStream()); Object resp = processRequest(req); if(resp != null){ xstream.toXML(resp, response.getOutputStream()); } else {// Check the null } } private Object processRequest(Object req){ if(req.getClass() == UploadRequest.class){ UploadRequest upload = (UploadRequest)req; return processUpload(upload); } else if (req.getClass() == HeartBeatRequest.class) { HeartBeatRequest heartBeat = (HeartBeatRequest)req; return processHeartBeat(heartBeat); } else { return null; } } /** * Just for fun * @param upload * @return */ private UploadResponse processUpload(UploadRequest upload) { //Just for tracking the message flow Message message = new Message(); message.setFrom(upload.getId()); message.setDate(upload.getDate()); message.setContent(upload.getContent()); map.add(upload.getTo(), message); UploadResponse resp = null; if(true/*upload.getId().equals(new Integer(1))&&upload.getPwd().equals("19841230")*/){ resp = new UploadResponse(); resp.setFlag(true); } else { resp = new UploadResponse(); resp.setFlag(false); } return resp; } /** * Just for fun * @param heartBeat * @return */ private HeartBeatResponse processHeartBeat(HeartBeatRequest heartBeat) { //Just for tracking the message flow HeartBeatResponse resp = new HeartBeatResponse(); //need to do some login process if(true/*heartBeat.getId().equals(new Integer(1))&&heartBeat.getPwd().equals("19841230")*/){ resp.setFlag(true); List<Message> list = map.getMessage(heartBeat.getId()); resp.setMessages(list); } else { resp.setFlag(false); } return resp; } }
写完了。好累。
最后写下怎么用吧:在ChatClient中修改usrId、distId成员来确定发送者和接受者。设成相反的两个数字就可以,当然,客户端要启动2次。验证功能目前是关闭的。password无所谓了。
MessagePoster中的url可能需要修改一下,如果你部署的时候改变路径了。
就这么多了
- HttpChat.rar (855.8 KB)
- 下载次数: 108
评论
4 楼
tianzhihehe
2009-12-16
支持,我一直在想用什么协议来传输数据呢。看了楼主的文章,应该是SOAP了。拜读……
3 楼
Aga
2008-08-19
Zhang Joanna@prysmian.com
2 楼
sion
2008-08-09
gmail
1 楼
jelver
2008-08-09
感觉还不错,看看
发表评论
-
ant build.xml例子
2010-01-10 20:16 2500<?xml version="1.0" ... -
避免内存泄露专题
2009-12-17 16:19 1217For Java: 1.限制使用单例模式;单例模式是引发mem ... -
万恶的inner class, memory leak的源头之一
2009-12-17 15:35 1316非static inner class隐式的含有一个引用指向o ... -
resolve maven [Request to merge when 'filtering' is not identical]. issue
2009-11-06 11:18 2694mvn org.apache.maven.plugins:ma ... -
an example of cloneable
2009-10-10 09:43 1075The following code describes ho ... -
jvm 远程debug
2009-04-17 10:01 1132-Xdebug -Xrunjdwp:transport=dt_ ... -
java dynamic proxy
2008-11-21 20:45 1126package proxy.cxz.org; impor ... -
serializable例子一则
2008-11-15 21:13 1047实现Serializable接口,编写地定义的针对transi ... -
jmx添加naming service以及一个rmi 监听方式
2008-11-11 15:22 2624ObjectName namingName = new ... -
指定编码器、解码器,并且利用ByteBuffer读写文件
2008-11-04 20:48 2683指定编码器、解码器,并且利用ByteBuffer读写文件。 做 ... -
java 网络编程探讨
2008-11-02 19:30 2092毕业有一年多了,大学的时候上过网络课程,但我一直认为:网络技术 ... -
Differences Between notify() and notifyAll()
2008-08-08 21:20 1043package com.cxz.currency.test; ... -
jdk1.6中添加的future
2008-07-18 13:59 1321摘自:http://caterpillar.onlyfun.n ... -
模拟jdk1.5中reentrantlock
2008-07-18 13:15 1197选自:java线程2e。 类似于jdk1.5中的reentra ... -
java线程2e中写得相当花哨的thread例子!
2008-07-14 16:53 1777太鲜灵儿了! package com.cxz.tools; ... -
j2se中实现jndi的控制、管理
2008-07-01 19:34 1154jndi例子一则,转载自网络。利用一个container实现了 ... -
jmx控制tomcat
2008-07-01 09:22 5044以前所作的一切应用程序基本上都是由tomcat容器,控制web ... -
jmx例子一则
2008-06-30 07:32 2882很简单的一个hellojmx的例子。其中实现了:标准bean、 ... -
nio与io的比较
2008-06-21 17:31 14933nio是new io的简称,从jdk1.4就被引入了。现在的j ... -
RTTI+Reflection总结
2008-06-19 09:54 1108package com.cxz.thinking.act1 ...
相关推荐
《基于Java的B/S结构人力资源管理系统实现》 在信息技术飞速发展的今天,企业对人力资源管理的需求日益增强,而基于B/S(Browser/Server)架构的人力资源管理系统因其便捷性、高效性和可扩展性,成为了企业管理的...
《基于Java,JSP,MySQL的医疗管理系统C/S架构详解》 在当今信息化时代,医疗管理系统已经成为医疗机构不可或缺的一部分。本文将深入探讨一个基于Java、JSP、MySQL技术的C/S(客户端/服务器)架构的医疗管理系统,...
【标题】:“基于B/S结构的交友网站” 这个标题揭示了我们要讨论的是一个采用Browser/Server(B/S)架构设计的在线社交平台。在B/S架构中,用户通过浏览器访问服务器上的应用程序,实现数据交互,这使得用户无需...
开发工具 apache-tomcat-8.0.41-windows-x86开发工具 apache-tomcat-8.0.41-windows-x86开发工具 apache-tomcat-8.0.41-windows-x86开发工具 apache-tomcat-8.0.41-windows-x86开发工具 apache-tomcat-8.0.41-...
数据库量大,功能很全的通讯录!此通讯录为b/s结构的 运行此系统的前提是: 安装jdk1.5 安装tomcat6.0服务器 建源程序复制到Tomcat 6.0\webapps\下 安装MySQL数据库——将建表语句导入数据库
如果,有一种C/S结构的网络中间件,独立承担网络和安全性等的处理,并且能如同Tomcat那样实现开发过程中代码的热部署,那 么,就能像Tomcat一样,为全世界的C/S结构系统的开发带来低成本,高效率和高稳定性了。...
本资源包含一个 openssl 工具安装包 Win32OpenSSL-1_1_0c.exe,一个 tomcat 进行配置 ssl 证书、完全 TLS v1.2、完全正向加密的 server.xml、startup.bat 配置文件。关于tomcat 进行配置 ssl 证书、完全 TLS v1.2、...
标题中的"webServer tomcat5 / tomcat6 / tomcat7 / tomcat8"涉及到的是Apache Tomcat服务器的不同版本。Tomcat是一款开源的Java Servlet容器,主要用于实现Java Web应用程序的运行环境。它支持Servlet和JSP标准,由...
【标题】:Tomcat启动管理工具 在Java Web开发领域,Tomcat是一个广泛使用的开源应用服务器,主要用于部署和运行Servlet和JSP应用。Tomcat以其轻量级、易配置和高性能的特点深受开发者喜爱。"Tomcat启动管理工具"指...
Tomcat是符合Java Servlet和JavaServer Pages(JSP)规范的应用服务器,它是基于Java的开源软件,主要用于处理动态网页内容。Tomcat不仅免费,而且易于配置和管理,因此在小型项目或开发环境中特别受欢迎。 2. ...
首先,系统采用B/S模式作为学生端,这是因为B/S模式基于Web浏览器,用户只需打开网页即可进行操作,无需安装额外软件,易于部署和使用。而C/S模式则用于教师端,它利用Java Swing技术构建客户端应用程序,提供更丰富...
JavaServer Pages(JSP)则是一种动态网页技术,它将HTML代码与Java代码分离,使开发者能更方便地创建基于Java的交互式网页。Tomcat作为Servlet和JSP容器,提供了这些技术的运行环境。 Tomcat 6.0.26 版本是在2009...
Apache Tomcat 7.0.12 是一个广泛使用的开源软件,它是一个实现了Java Servlet、JavaServer Pages(JSP)和Java EE的Web应用程序容器。这个版本是专为64位Windows操作系统设计的,提供了无需安装的绿色版本,使得...
在这个项目中,“基于Tomcat8+Java7+ExtJS的WebSocket聊天室实现”利用了这些技术来创建一个实时交互的聊天应用。 1. **WebSocket简介** WebSocket为Web应用提供了低延迟、双向通信的能力,使得服务器能够主动向...
【Ajax与Servlet在网页聊天工具中的应用】 Ajax(Asynchronous JavaScript and XML)是一种在无需刷新整个网页的情况下,能够更新部分网页的技术。它通过JavaScript发送异步HTTP请求到服务器,然后在后台处理数据并...
### Ubuntu中部署Tomcat知识点详解 #### 一、前言 在现代Web开发环境中,Apache Tomcat是一款广泛使用的开源Java Servlet容器。它不仅轻量级且功能强大,能够支持多种应用部署需求。对于使用Ubuntu系统的开发者来...
### Servlet与JSP深入详解:基于Tomcat的Web开发 #### 一、Servlet技术概述 Servlet是一种用Java编写的服务器端应用程序接口(API),它扩展了Java Web开发的功能。Servlet可以响应HTTP请求,并且能够处理复杂的...
**标题:“Tomcat-9.0-API”** **概述:** Tomcat-9.0-API 是针对Apache Tomcat 9.0版本的官方应用程序接口(API)文档,它提供了开发者们详细的技术指南,用于理解并使用Tomcat服务器的核心功能。Tomcat是一个...
这个版本是Tomcat服务器的一个重要分支,它提供了对Java技术的高效且轻量级的支持,使得开发者能够部署和运行基于Java的Web应用。 在Ubuntu操作系统上安装Apache Tomcat 8.0.20,首先你需要确保系统已经安装了Java ...