目录

  前言

  后端

  浏览器前端

java的client

  注意

前言

HTML5 WebSocket实现了服务器与浏览器的双向通讯,开销小,实时性高,常用于即时通讯和对信息实时性要求比较高的应用。
下面讲解如利用Tomcat8+WebSocket开发聊天室。
目前Spring已经推出了WebSocket的API,能够兼容各个服务器的实现

后端

 说明

@ServerEndpoint("/echo") 的 annotation 注释端点表示将 WebSocket 服务端运行在 ws://[Server 端 IP 或域名]:[Server 端口]/websockets/echo 的访问端点,客户端浏览器已经可以对 WebSocket 客户端 API 发起 HTTP 长连接了。使用 ServerEndpoint 注释的类必须有一个公共的无参数构造函数,
@onMessage 注解的 Java 方法用于接收传入的 WebSocket 信息,这个信息可以是文本格式,可以是json数据,也可以是二进制格式。
@OnOpen 在这个端点一个新的连接建立时被调用。参数提供了连接的另一端的更多细节。Session 表明两个 WebSocket 端点对话连接的另一端,可以理解为类似 HTTPSession 的概念。
@OnClose 在连接被终止时调用。参数 closeReason 可封装更多细节,如为什么一个 WebSocket 连接关闭。
@Message 注释,MaxMessageSize 属性可以被用来定义消息字节最大限制,在示例程序中,如果超过 6 个字节的信息被接收,就报告错误和连接关闭。

   ChatServer.java

import java.io.IOException;
import java.util.concurrent.CopyOnWriteArraySet;
import javax.websocket.OnClose;
import javax.websocket.OnError;
import javax.websocket.OnMessage;
import javax.websocket.OnOpen;
import javax.websocket.Session;
import javax.websocket.server.PathParam;
import javax.websocket.server.ServerEndpoint;

//该注解用来指定一个URI,客户端可以通过这个URI来连接到WebSocket。类似Servlet的注解mapping。无需在web.xml中配置。
@ServerEndpoint("/ws/chat/{nickName}")
public class ChatServer {
  //静态变量,用来记录当前在线连接数。应该把它设计成线程安全的。
  private static int onlineCount = 0;
  //concurrent包的线程安全Set,用来存放每个客户端对应的MyWebSocket对象。若要实现服务端与单一客户端通信的话,可以使用Map来存放,其中Key可以为用户标识
  private static CopyOnWriteArraySet<ChatServer> webSocketSet = new CopyOnWriteArraySet<ChatServer>();
  //与某个客户端的连接会话,需要通过它来给客户端发送数据
  private Session session;
  private String nickName;
  /**
  * 连接建立成功调用的方法
  * @param session 可选的参数。session为与某个客户端的连接会话,需要通过它来给客户端发送数据
  */
  @OnOpen
  public void onOpen(Session session, @PathParam(value = "nickName") String nickName){
    this.session = session;
    this.nickName = nickName;
    webSocketSet.add(this); //加入set中
    String message = String.format("System> %s %s", this.nickName, " 加入.");
    broadCast(message);
    addOnlineCount(); //在线数加1
    System.out.println(nickName+"新加入!当前在线人数为" + getOnlineCount());
  }

  /**
  * 连接关闭调用的方法
  */
  @OnClose
  public void onClose(){
    webSocketSet.remove(this);   //从set中删除
    subOnlineCount();         //在线数减1
    String message = String.format("System> %s, %s", this.nickName, " 离开.");
    broadCast(message);
    System.out.println(this.nickName+"离开!当前在线人数为" + getOnlineCount());
  }

  /**
  * 收到客户端消息后调用的方法
  * @param message 客户端发送过来的消息
  * @param session 可选的参数
  */
  @OnMessage
  public void onMessage(String message, Session session, @PathParam(value = "nickName") String nickName) {
    System.out.println(nickName+":" + message);
    broadCast(nickName + ">" + message);
  }

  /**
  * 发生错误时调用
  * @param session
  * @param error
  */
  @OnError
  public void onError(Session session, Throwable throwable) {
    System.out.println("发生错误");
    System.out.println(throwable.getMessage());
  }

  /**
  * 发送或广播信息
  * @param message
  */
  private void broadCast(String message) {
    //群发消息
    for (ChatServer chat : webSocketSet) {
    try {
      synchronized (chat) {
        chat.session.getBasicRemote().sendText(message);
      }
    } catch (IOException e) {
      webSocketSet.remove(chat);
      try {
          chat.session.close();
        } catch (IOException e1) {
      }
        broadCast(String.format(" %s %s", chat.nickName, " has bean disconnection."));
      }
    }
  }

  public static synchronized int getOnlineCount() {
    return onlineCount;
  }

  public static synchronized void addOnlineCount() {
    ChatServer.onlineCount++;
  }

  public static synchronized void subOnlineCount() {
    ChatServer.onlineCount--;
  }
}

前端

  chat.jsp 

<%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%>
<%
   String path =
request.getContextPath();
   String basePath =
request.getScheme() + "://" + request.getServerName() + ":" + request.getServerPort() + path + "/";
   String nickName=request.getParameter("nickName");
%>
<!DOCTYPE HTML>
<html>
<head>
   <base href="<%=basePath%>">
   <title>My WebSocket</title>
</head>
<body>
   Welcome   <br />
   <div id="message"></div> 
   <input id="text" type="text" />
   <button onclick="send()">Send</button>
   <button onclick="closeWebSocket()">Close</button>
</body> 
<script type="text/javascript">
   var websocket = null;
   //判断当前浏览器是否支持WebSocket
   if ('WebSocket'in window) {
      websocket = new
WebSocket("ws://localhost:8080/webSocket/ws/chat/<%=nickName%>");
   } else {
      alert('Not support websocket')
   }
   //连接发生错误的回调方法
   websocket.onerror = function() {
      setMessageInnerHTML("error");
   };
   //连接成功建立的回调方法
   websocket.onopen = function(event) {
      setMessageInnerHTML("open");
   }
   //接收到消息的回调方法
   websocket.onmessage = function() {
      setMessageInnerHTML(event.data);
   }
   //连接关闭的回调方法
   websocket.onclose = function() {
      setMessageInnerHTML("close");
   }
   //监听窗口关闭事件,当窗口关闭时,主动去关闭websocket连接,防止连接还没断开就关闭窗口,server端会抛异常。
   window.onbeforeunload = function() {
      websocket.close();
   }
   //将消息显示在网页上
   function setMessageInnerHTML(innerHTML) {
      document.getElementById('message').innerHTML += innerHTML + '<br/>';
   } 
   //关闭连接
   function closeWebSocket() {
      websocket.close();
   } 
   //发送消息
   function send() {
      var message =
document.getElementById('text').value;
      websocket.send(message);
   }
</script> 
</html>

访问

http://127.0.0.1:8080/webSocket/chat.jsp?nickName=dengzy

http://127.0.0.1:8080/webSocket/chat.jsp?nickName=dengbw

java的client

  java_websocket.

import java.net.URI;

import
java.net.URISyntaxException;

import java.util.Scanner;

import
org.java_websocket.client.WebSocketClient;

import
org.java_websocket.drafts.Draft_17;

import
org.java_websocket.handshake.ServerHandshake;

publicclass TestTocatWebSocket
{

publicstaticvoid main(String[] args) throws URISyntaxException
{

String url = "ws://localhost:8080/webSocket/ws/chat/"+ args[0] ;//

WebSocketClient
wc = new WebSocketClient(new URI(url), new Draft_17()) {

@Override

publicvoid
onOpen(ServerHandshake handshakedata) {

System.out.println(handshakedata.getHttpStatusMessage());

}

@Override

publicvoid onMessage(String message) {

System.out.println(message);

}

@Override

publicvoid onError(Exception ex) {

}

@Override

publicvoid onClose(intcode, String reason, booleanremote) {

}

};

wc.connect();

while (true) {

Scanner scanner = new Scanner(System.in);

String message = scanner.nextLine();

if (message.equals("q")) {

wc.close();

break;

}

scanner.close();

wc.send(message);

}

}

}

注意

 1.在部署完成之后需要将在tomcat应用目录中的lib目录下的catalina.jar和tomcat-coyote.jar删掉,比如项目的lib目录在D:\workspace\WebSocket\WebRoot\WEB-INF\lib,而部署的应用lib目录是在D:\tools\apache-tomcat-7.0.32\webapps\WebSocket\WEB-INF\lib,删掉部署目录的lib目录中连两个jar就可以了,否则会包Could not initialize class com.ibcio.WebSocketMessageServlet错误。

 2.Tomcat8安装包里有WebSocket的demo

3.html5 才支持WebSocket 

html5_websocket_tomcat8的更多相关文章

随机推荐

  1. 【旋转卡壳】poj3608 Bridge Across Islands

    给你俩凸包,问你它们的最短距离. 咋做就不讲了,经典题,网上一片题解. 把凸包上的点逆时针排序.可以取它们的平均点,然后作极角排序. 旋转卡壳其实是个很模板化的东西…… 先初始化分别在凸包P和Q上取哪 ...

  2. es 数据 导出 到 MySQL

    暂时没有找到直接 导出到 mysql 数据库的工具 或者项目 目前实现思路: 使用 elasticdump  工具 实现 从 es 数据 导出到 json 文件 ,然后 使用 脚本程序 操作 改 js ...

  3. openresty总结

    协程 1.例如当获取的数据没有前后依赖关系时,可以使用ngx.thread.spawn和ngx.thread.wait同时从数据库不同的库.表或者不同来源(mysql,redis等)获取数据. htt ...

  4. 扩展gridview轻松实现冻结行和列(增强型)

    上一篇说过,还可以扩展gridview的分页功能以及实现导出结果为EXCEL/PDF的功能.实现好后应该封装起来,以方便后续的项目简单使用.至于要如何实现,我想不必过多的说了.下面是显示结果和主要的代 ...

  5. java工具类获取properties文件的配置

    import java.io.BufferedReader; import java.io.IOException; import java.io.InputStream; import java.i ...

  6. 解决Windows服务修改配置文件后必须重启的问题

      原文地址:http://www.cnblogs.com/jeffwongishandsome/archive/2011/04/24/2026381.html   解决方法:读取配置文件前先刷新文件 ...

  7. XSS跨站脚本测试用例

    '><script>alert(document.cookie)</script>='><script>alert(document.cookie)&l ...

  8. javascript快速入门15--节点

    节点类型 DOM定义了Node的接口以及许多种节点类型来表示节点的多个方面! Document——最顶层的节点,所有的其他节点都是附属于它的. DocumentType——DTD引用(使用<!D ...

  9. linux的chmod命令

    chmod命令用来变更文件或目录的权限.在UNIX系统家族里,文件或目录权限的控制分别以读取.写入.执行3种一般权限来区分,另有3种特殊权限可供运用.用户可以使用chmod指令去变更文件与目录的权限, ...

  10. C# 关键字 Visual Studio 2012

    C# 关键字 Visual Studio 2012 其他版本 关键字是对编译器具有特殊意义的预定义保留标识符. 它们不能在程序中用作标识符,除非它们有一个 @ 前缀. 例如,@if 是有效的标识符,但 ...