`
Aga
  • 浏览: 217725 次
  • 性别: Icon_minigender_1
  • 来自: 天津
社区版块
存档分类
最新评论

一套基于http的聊天c/s结构工具(除了网页tomcat还能做什么)

    博客分类:
  • J2SE
阅读更多
在我的认识当中以前一直有一种误区认为: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端使用具体代码如下:

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可能需要修改一下,如果你部署的时候改变路径了。

就这么多了
7
3
分享到:
评论
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  
感觉还不错,看看

相关推荐

    基于Java的B/S结构的人力资源管理系统实现+java源码+Sql数据+页面

    《基于Java的B/S结构人力资源管理系统实现》 在信息技术飞速发展的今天,企业对人力资源管理的需求日益增强,而基于B/S(Browser/Server)架构的人力资源管理系统因其便捷性、高效性和可扩展性,成为了企业管理的...

    基于java,jsp,mysql的医疗管理系统C/S结构

    《基于Java,JSP,MySQL的医疗管理系统C/S架构详解》 在当今信息化时代,医疗管理系统已经成为医疗机构不可或缺的一部分。本文将深入探讨一个基于Java、JSP、MySQL技术的C/S(客户端/服务器)架构的医疗管理系统,...

    基于B/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-windows-x86开发工具 apache-tomcat-8.0.41-...

    B/S结构 (Java )通讯录

    数据库量大,功能很全的通讯录!此通讯录为b/s结构的 运行此系统的前提是: 安装jdk1.5 安装tomcat6.0服务器 建源程序复制到Tomcat 6.0\webapps\下 安装MySQL数据库——将建表语句导入数据库

    AnyFo - EricSofa:专门用于C/S结构的网络服务器中间件

    如果,有一种C/S结构的网络中间件,独立承担网络和安全性等的处理,并且能如同Tomcat那样实现开发过程中代码的热部署,那 么,就能像Tomcat一样,为全世界的C/S结构系统的开发带来低成本,高效率和高稳定性了。...

    SSL/TLS 检测工具以及 tomcat 正向加密配置例子

    本资源包含一个 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

    标题中的"webServer tomcat5 / tomcat6 / tomcat7 / tomcat8"涉及到的是Apache Tomcat服务器的不同版本。Tomcat是一款开源的Java Servlet容器,主要用于实现Java Web应用程序的运行环境。它支持Servlet和JSP标准,由...

    tomcat启动管理工具

    【标题】:Tomcat启动管理工具 在Java Web开发领域,Tomcat是一个广泛使用的开源应用服务器,主要用于部署和运行Servlet和JSP应用。Tomcat以其轻量级、易配置和高性能的特点深受开发者喜爱。"Tomcat启动管理工具"指...

    tomcat工具

    Tomcat是符合Java Servlet和JavaServer Pages(JSP)规范的应用服务器,它是基于Java的开源软件,主要用于处理动态网页内容。Tomcat不仅免费,而且易于配置和管理,因此在小型项目或开发环境中特别受欢迎。 2. ...

    java学生成绩管理系统B/S+C/S

    首先,系统采用B/S模式作为学生端,这是因为B/S模式基于Web浏览器,用户只需打开网页即可进行操作,无需安装额外软件,易于部署和使用。而C/S模式则用于教师端,它利用Java Swing技术构建客户端应用程序,提供更丰富...

    apache-tomcat-6.0.26

    JavaServer Pages(JSP)则是一种动态网页技术,它将HTML代码与Java代码分离,使开发者能更方便地创建基于Java的交互式网页。Tomcat作为Servlet和JSP容器,提供了这些技术的运行环境。 Tomcat 6.0.26 版本是在2009...

    apache-tomcat-7.0.12-windows-x64.zip

    Apache Tomcat 7.0.12 是一个广泛使用的开源软件,它是一个实现了Java Servlet、JavaServer Pages(JSP)和Java EE的Web应用程序容器。这个版本是专为64位Windows操作系统设计的,提供了无需安装的绿色版本,使得...

    基于tomcat8+java7+extjs 的webscoket 聊天室实现

    在这个项目中,“基于Tomcat8+Java7+ExtJS的WebSocket聊天室实现”利用了这些技术来创建一个实时交互的聊天应用。 1. **WebSocket简介** WebSocket为Web应用提供了低延迟、双向通信的能力,使得服务器能够主动向...

    基于Ajax和servlet的网页聊天工具

    【Ajax与Servlet在网页聊天工具中的应用】 Ajax(Asynchronous JavaScript and XML)是一种在无需刷新整个网页的情况下,能够更新部分网页的技术。它通过JavaScript发送异步HTTP请求到服务器,然后在后台处理数据并...

    ubuntu中部署tomcat

    ### Ubuntu中部署Tomcat知识点详解 #### 一、前言 在现代Web开发环境中,Apache Tomcat是一款广泛使用的开源Java Servlet容器。它不仅轻量级且功能强大,能够支持多种应用部署需求。对于使用Ubuntu系统的开发者来...

    Servlet JSP深入详解 基于Tomcat的Web开发

    ### Servlet与JSP深入详解:基于Tomcat的Web开发 #### 一、Servlet技术概述 Servlet是一种用Java编写的服务器端应用程序接口(API),它扩展了Java Web开发的功能。Servlet可以响应HTTP请求,并且能够处理复杂的...

    Tomcat-9.0-API

    **标题:“Tomcat-9.0-API”** **概述:** Tomcat-9.0-API 是针对Apache Tomcat 9.0版本的官方应用程序接口(API)文档,它提供了开发者们详细的技术指南,用于理解并使用Tomcat服务器的核心功能。Tomcat是一个...

    apache-tomcat-8.0.20

    这个版本是Tomcat服务器的一个重要分支,它提供了对Java技术的高效且轻量级的支持,使得开发者能够部署和运行基于Java的Web应用。 在Ubuntu操作系统上安装Apache Tomcat 8.0.20,首先你需要确保系统已经安装了Java ...

Global site tag (gtag.js) - Google Analytics