zoukankan      html  css  js  c++  java
  • Java开发之使用websocket实现web客户端与服务器之间的实时通讯

    使用websocket实现web客户端与服务器之间的实时通讯。以下是个简单的demo。

    前端页面

      1 <%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%>
      2 <!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
      3 <%@ taglib uri="http://java.sun.com/jsp/jstl/core"  prefix="c"%>
      4 <%@ taglib uri="http://java.sun.com/jsp/jstl/fmt"  prefix="fmt"%>
      5 <%@taglib uri="http://java.sun.com/jsp/jstl/functions" prefix="fun"%>
      6 <%@ taglib prefix="fn" uri="http://java.sun.com/jsp/jstl/functions"%> 
      7 <c:set var="baseurl" value="${pageContext.request.contextPath}/"></c:set>
      8 
      9 <html>
     10 <head>
     11 <meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
     12 <title>web_socket</title>
     13 
     14 <script type="text/javascript" src="${baseurl}static/js/jquery-2.1.1.js"></script>
     15 <style type="text/css">
     16     .connector {width: 500px;}
     17 </style>
     18 </head>
     19 <body>
     20     Welcome
     21     <br/>
     22     <input id="text" type="text"/>
     23     <button onclick="sendToOne(10008)">发消息给个人</button>
     24     <button onclick="sendToAll(0)">发消息给所有人</button>
     25     <hr/>
     26     <button onclick="closeWebSocket()">关闭WebSocket连接</button>
     27     <hr/>
     28     <div id="message"></div>
     29 </body>
     30 <script type="text/javascript">
     31     var websocket = null;
     32     var host = document.location.host;
     33     
     34     //判断当前浏览器是否支持WebSocket
     35     if ('WebSocket' in window) {
     36         console.info("浏览器支持Websocket");
     37         websocket = new WebSocket('ws://'+host+'/${baseurl}/webSocketServer/${userID}');
     38     } else {
     39         console.info('当前浏览器 Not support websocket');
     40     }
     41     
     42     //连接发生错误的回调方法
     43     websocket.onerror = function() {
     44         console.info("WebSocket连接发生错误");
     45         setMessageInnerHTML("WebSocket连接发生错误"); 
     46     }
     47     
     48     //连接成功建立的回调方法
     49     websocket.onopen = function() {
     50         console.info("WebSocket连接成功");
     51         setMessageInnerHTML("WebSocket连接成功"); 
     52     } 
     53     
     54     //接收到消息的回调方法 
     55     websocket.onmessage = function(event) {
     56         console.info("接收到消息的回调方法");
     57         console.info("这是后台推送的消息:"+event.data);
     58         setMessageInnerHTML(event.data); 
     59         console.info("webSocket已关闭!");
     60     }
     61     
     62     //连接关闭的回调方法 
     63     websocket.onclose = function() {
     64         setMessageInnerHTML("WebSocket连接关闭");
     65     }
     66     
     67     //监听窗口关闭事件,当窗口关闭时,主动去关闭websocket连接,防止连接还没断开就关闭窗口,server端会抛异常。
     68     window.onbeforeunload = function() {
     69         closeWebSocket();
     70     }
     71     
     72     //关闭WebSocket连接 
     73     function closeWebSocket() {
     74         websocket.close();
     75     }
     76     
     77     //将消息显示在网页上
     78     function setMessageInnerHTML(innerHTML) {
     79         document.getElementById('message').innerHTML += innerHTML + '<br/>';
     80     }
     81     
     82     //发送消息给其他客户端
     83     function sendToOne(receiverId) {
     84         var messageContent = document.getElementById('text').value;
     85         var message = {};
     86         message.senderId = "${userID}";
     87         message.receiverId = receiverId;
     88         message.messageContent = messageContent;
     89         websocket.send(JSON.stringify(message));
     90     }
     91     
     92     //发送消息给所有人
     93     function sendToAll() {
     94         var messageContent = document.getElementById('text').value;
     95         var message = {};
     96         message.senderId = "${userID}";
     97         message.receiverId = "0";
     98         message.messageContent = messageContent;
     99         websocket.send(JSON.stringify(message));
    100     }
    101 </script>
    102 </html>

    后台代码

     1 import java.util.Date;
     2 
     3 public class WebSocketMessage {
     4 
     5     /**
     6      * 发送者ID
     7      */
     8     private String senderId;
     9 
    10     /**
    11      * 接受者ID, 如果为0, 则发送给所有人
    12      */
    13     private String receiverId;
    14 
    15     /**
    16      * 会话内容
    17      */
    18     private String messageContent;
    19 
    20     /**
    21      * 发送时间
    22      */
    23     private Date sendTime;
    24 
    25     public String getSenderId() {
    26         return senderId;
    27     }
    28 
    29     public void setSenderId(String senderId) {
    30         this.senderId = senderId;
    31     }
    32 
    33     public String getReceiverId() {
    34         return receiverId;
    35     }
    36 
    37     public void setReceiverId(String receiverId) {
    38         this.receiverId = receiverId;
    39     }
    40 
    41     public String getMessageContent() {
    42         return messageContent;
    43     }
    44 
    45     public void setMessageContent(String messageContent) {
    46         this.messageContent = messageContent;
    47     }
    48 
    49     public Date getSendTime() {
    50         return sendTime;
    51     }
    52 
    53     public void setSendTime(Date sendTime) {
    54         this.sendTime = sendTime;
    55     }
    56 
    57 }
     1 import javax.servlet.http.HttpServletResponse;
     2 
     3 import org.springframework.stereotype.Controller;
     4 import org.springframework.web.bind.annotation.PathVariable;
     5 import org.springframework.web.bind.annotation.RequestMapping;
     6 import org.springframework.web.servlet.ModelAndView;
     7 
     8 @Controller
     9 @RequestMapping("webSocket")
    10 public class WebSocketController {
    11 
    12     @RequestMapping(value = "messagePage/{userID}")
    13     public ModelAndView messagePage(@PathVariable String userID, HttpServletResponse response) {
    14         ModelAndView mav = new ModelAndView();
    15         mav.addObject("userID", userID);
    16         mav.setViewName("web_socket");
    17         return mav;
    18     }
    19 }
      1 import java.io.IOException;
      2 import java.util.Map;
      3 import java.util.concurrent.ConcurrentHashMap;
      4 
      5 import javax.websocket.OnClose;
      6 import javax.websocket.OnError;
      7 import javax.websocket.OnMessage;
      8 import javax.websocket.OnOpen;
      9 import javax.websocket.Session;
     10 import javax.websocket.server.PathParam;
     11 import javax.websocket.server.ServerEndpoint;
     12 
     13 import com.alibaba.fastjson.JSON;
     14 import com.utime.facade.model.systemplate.WebSocketMessage;
     15 
     16 @ServerEndpoint("/webSocketServer/{userID}")
     17 public class WebSocketServer {
     18     // 连接客户端数量
     19     private static int onlineCount = 0;
     20     // 所有的连接客户端
     21     private static Map<String, WebSocketServer> clients = new ConcurrentHashMap<String, WebSocketServer>();
     22     // 当前客户端连接的唯一标示
     23     private Session session;
     24     // 当前客户端连接的用户ID
     25     private String userID;
     26 
     27     /**
     28      * 客户端连接服务端回调函数
     29      * 
     30      * @param userID 用户ID
     31      * @param session 会话
     32      * @throws IOException
     33      */
     34     @OnOpen
     35     public void onOpen(@PathParam("userID") String userID, Session session) throws IOException {
     36         this.userID = userID;
     37         this.session = session;
     38 
     39         addOnlineCount();
     40         clients.put(userID, this);
     41         System.out.println("WebSocket日志: 有新连接加入!当前在线人数为" + getOnlineCount());
     42     }
     43 
     44     @OnClose
     45     public void onClose() throws IOException {
     46         clients.remove(userID);
     47         subOnlineCount();
     48         System.out.println("WebSocket日志: 有一连接关闭!当前在线人数为" + getOnlineCount());
     49     }
     50 
     51     /**
     52      * 接受到来自客户端的消息
     53      * 
     54      * @param message
     55      * @throws IOException
     56      */
     57     @OnMessage
     58     public void onMessage(String message) throws IOException {
     59         System.out.println("WebSocket日志: 来自客户端的消息:" + message);
     60         WebSocketMessage webSocketMessage = JSON.parseObject(message, WebSocketMessage.class);
     61 
     62         // 发送消息给所有客户端
     63         if ("0".equals(webSocketMessage.getReceiverId())) {
     64             for (WebSocketServer item : clients.values()) {
     65                 item.session.getAsyncRemote().sendText(webSocketMessage.getMessageContent());
     66                 System.out.println("WebSocket日志: ID为"+ webSocketMessage.getSenderId() +"的用户给ID为"+ item.userID +"的客户端发送:" + webSocketMessage.getMessageContent());
     67             }
     68         } else {    // 发送消息给指定ID的客户端
     69             for (WebSocketServer item : clients.values()) {
     70                 if (item.userID.equals(webSocketMessage.getReceiverId())){
     71                     // 发消息给指定客户端
     72                     item.session.getAsyncRemote().sendText(webSocketMessage.getMessageContent());
     73                     System.out.println("WebSocket日志: ID为"+ webSocketMessage.getSenderId() +"的用户给ID为"+ item.userID +"的客户端发送:" + webSocketMessage.getMessageContent());
     74                     if (!webSocketMessage.getSenderId().equals(webSocketMessage.getReceiverId())) {
     75                         // 发消息给自己
     76                         this.session.getAsyncRemote().sendText(webSocketMessage.getMessageContent());
     77                         System.out.println("WebSocket日志: ID为"+ webSocketMessage.getSenderId() +"的用户给ID为"+ this.userID +"的客户端发送:" + webSocketMessage.getMessageContent());
     78                     }
     79                     break;
     80                 }
     81             }
     82         }
     83     }
     84     
     85     /**
     86      * 服务端报错了
     87      * 
     88      * @param session
     89      * @param error
     90      */
     91     @OnError
     92     public void onError(Session session, Throwable error) {
     93         System.out.println("WebSocket日志: 发生错误");
     94         error.printStackTrace();
     95     }
     96     
     97     /**
     98      * 客户端连接数+1
     99      */
    100     public static synchronized void addOnlineCount() {
    101         WebSocketServer.onlineCount++;
    102     }
    103     
    104     /**
    105      * 客户端连接数-1
    106      */
    107     public static synchronized void subOnlineCount() {
    108         WebSocketServer.onlineCount--;
    109     }
    110 
    111     public static synchronized int getOnlineCount() {
    112         return onlineCount;
    113     }
    114 
    115     public static synchronized Map<String, WebSocketServer> getClients() {
    116         return clients;
    117     }
    118 }

    写这个的目的只是为了自己做个记录。

  • 相关阅读:
    网络服务—VSFTP
    DHCP服务基本搭建
    zabbix使用钉钉告警
    差分数组
    最小点权覆盖集&最大点权独立集
    康托展开&康托逆展开 的写法
    可并堆(左偏树)简单学习
    树链剖分原理与应用
    后缀数组学习笔记
    HDU-3974 Assign the task题解报告【dfs序+线段树】
  • 原文地址:https://www.cnblogs.com/zhaosq/p/11804154.html
Copyright © 2011-2022 走看看