zoukankan      html  css  js  c++  java
  • websocket(二)--简单实现网页版群聊

    websocket能够实现服务端的消息推送。而不必在client轮询。大大的节省的资源,对于实时通讯来说简直是个大喜讯。

    在上一篇文章中介绍了协议握手,这篇文章将通过实现简单的群聊来帮助进一步了解websocket。

    注意:1.JavaEE版本号为7.0

                 2.tomcat版本号为8.0 

                 3.不须要额外导入其它的jar包 

                 因为websocket不支持低版本号的javaEE和tomcat,所以必须符合上述条件。我是在Myeclipse2014 的IDE中进行编码的。

                 为了尽可能的使代码简洁以便更好的理解,所以代码中有些地方可能不全面,但不影响学习websocket。 

                 首先是WebSocketServer类:

                  

    package socket;
    
    import java.io.IOException;
    import java.util.Set;
    import java.util.concurrent.CopyOnWriteArraySet;
    
    import javax.servlet.http.HttpSession;
    import javax.websocket.EndpointConfig;
    import javax.websocket.OnClose;
    import javax.websocket.OnMessage;
    import javax.websocket.OnOpen;
    import javax.websocket.Session;
    import javax.websocket.server.ServerEndpoint;
    
    @ServerEndpoint(value = "/socket",configurator=GetHttpSessionConfigurator.class)  
    public class WebSocketServer {
    	
    	private static final Set<WebSocketServer> onlineUsers =
                new CopyOnWriteArraySet<WebSocketServer>();  //静态变量 存储在线用户
    	private HttpSession httpSession;
    	private Session session;
    	private String name;
    	
    	@OnMessage  
        public void onMessage(String message, Session session)   
            throws IOException, InterruptedException { 
    	
    		//将client传来的消息发送给全部在线用户
    		sendMessageToAllOnline(session,this.name+" : "+message);
    		
    		//在控制台打印当前在线用户
    		for(WebSocketServer webSocketServer:onlineUsers){
            	System.out.print(webSocketServer.name+"  ");
            }
            System.out.println();
        }  
          
        @OnOpen  
        public void onOpen (Session session, EndpointConfig config) throws IOException {  
            System.out.println("Client connected");  
            
            this.session = session;
            this.httpSession = (HttpSession) config.getUserProperties()
                    .get(HttpSession.class.getName());
            this.name=(String) httpSession.getAttribute("name");
            onlineUsers.add(this);   //将用户信息加入到在线用户序列中
            
            //将上线消息发送给全部在线用户
            sendMessageToAllOnline(session,this.name+" 上线了");
            
        }  
      
        @OnClose  
        public void onClose(Session session) throws IOException{  
        	onlineUsers.remove(this);
            System.out.println("Connection closed"); 
            //将下线消息发送给全部在线用户
            sendMessageToAllOnline(session,this.name+" 下线了");
            
        }  
       /**
        * 发送信息给全部在线用户.
        * @param session
        * @param message
        */
       public void sendMessageToAllOnline(Session session,String message) throws IOException{
    	   for (Session sess : session.getOpenSessions()){
    	  		  if (sess.isOpen()){
    	  			  sess.getBasicRemote().sendText(message);			  
    	  		  }
    	   }
       }
    }
    
    因为这里边的session和平时使用的httpsession不是同一个,所以为了能够获取httpsession以便获取当前用户的信息,所以须要以下的类:

    package socket;
    
    import javax.servlet.http.HttpSession;
    import javax.websocket.HandshakeResponse;
    import javax.websocket.server.HandshakeRequest;
    import javax.websocket.server.ServerEndpointConfig;
    /**
     * 为获取httpSession提供支持
     * @author nagsh
     *
     */
    public class GetHttpSessionConfigurator extends ServerEndpointConfig.Configurator{
    	@Override
        public void modifyHandshake(ServerEndpointConfig config,
                                    HandshakeRequest request,
                                    HandshakeResponse response){
            HttpSession httpSession = (HttpSession)request.getHttpSession();
            config.getUserProperties().put(HttpSession.class.getName(),httpSession);
        }
    }
    

    该类在websocket的@ServerEndpoint注解中引用。

    为了能够获取登陆用户的信息以便实现群聊,简单的写了一个登陆,会将用户信息存到session中

    client:

    login.html

    <!DOCTYPE html>
    <html>
      <head>
        <title>login.html</title>
      </head>
      
      <body>
        <form action="servlet/LoginServlet" method="post">
                   姓名:<input type="text" name="name"/>
           <input type="submit">
        </form>
      </body>
    </html>
    

    服务端:

    package socket;
    
    import java.io.IOException;
    import java.io.PrintWriter;
    
    import javax.servlet.ServletException;
    import javax.servlet.http.HttpServlet;
    import javax.servlet.http.HttpServletRequest;
    import javax.servlet.http.HttpServletResponse;
    /**
     * 处理用户的登陆.
     * @author nagsh
     *
     */
    public class LoginServlet extends HttpServlet {
    
    	
    	public void doGet(HttpServletRequest request, HttpServletResponse response)
    			throws ServletException, IOException {
    		doPost(request, response);
    	}
    
    	
    	public void doPost(HttpServletRequest request, HttpServletResponse response)
    			throws ServletException, IOException {
    
    		response.setContentType("text/html");
    		PrintWriter out = response.getWriter();
    		
    		String name = request.getParameter("name");
    		//将当前用户的信息存入session中
    		request.getSession().setAttribute("name",name);
    		//重定向到聊天界面
    		response.sendRedirect("/websockettest/socketClient.jsp");
    	}
    
    }
    

    然后以下是简单的聊天界面(勿吐槽哈) 该页面是在上篇文章实现握手的页面基础上简单改动的:

    socketClient.jsp

    <%@ page language="java" import="java.util.*" pageEncoding="utf-8"%>
    
    <!DOCTYPE html>
    <html>
    <head>
    <title>Testing websockets</title>
     <script type="text/javascript" src="js/jquery.min.js"></script>
    </head>
    <script type="text/javascript">
    	    /*   申请一个WebSocket对象,參数是须要连接的服务器端的地址 */
    		var webSocket =  new WebSocket('ws://localhost:8080/websockettest/socket');
    
            /* 假设出现连接、处理、接收、发送数据失败的时候就会触发onerror事件 */
    		webSocket.onerror = function(event) {
    			onError(event)
    		};
    		
           /* 当websocket创建成功时,即会触发onopen事件 */
    		webSocket.onopen = function(event) {
    			onOpen(event)
    		};
    		
            /* 当client收到服务端发来的消息时,会触发onmessage事件,參数evt.data中包括server传输过来的数据 */
    		webSocket.onmessage = function(event) {
    			onMessage(event)
    		};
    		
            /* 当client收到服务端发送的关闭连接的请求时,触发onclose事件 */
            webSocket.onclose = function(event) {
    			onMessage(event)
    		};
    
    		function onMessage(event) {
    		   $("#messages").append(event.data+"<br/>");
    		}
    
    		function onOpen(event) {
    		    $("#messages").append("成功建立连接...<br/>");
    		    $("#messages").append(event.data);
    			
    		}
    
            function onClose(event) {
    			$("#messages").append(event.data);
    		}
    
    		function onError(event) {
    			alert("error");
    		}
    		
            //发送信息
            function send(){
                var content = $("#content").val();
                webSocket.send(content);      
            }
    	</script>
    <body>
    	
    	<div id="messages"></div>
    	<input type="text" id="content">
    	<input type="button" value="发送" onclick="send()">
    	
    </body>
    </html>





  • 相关阅读:
    物联网数据卡系统源码——物联网的主要应用领域
    一张图看懂开源许可协议,开源许可证GPL、BSD、MIT、Mozilla、Apache和LGPL的区别
    memcached对key和value的限制 memcached的key最大长度和Value最大长度
    缓存技术PK:选择Memcached还是Redis?
    .Net开源框架列表
    项目管理工具Redmine各功能测试
    DBImport V3.7版本发布及软件稳定性(自动退出问题)解决过程分享
    ASP.NET Core 折腾笔记一
    发布:.NET开发人员必备的可视化调试工具(你值的拥有)
    开源发布:VS代码段快捷方式及可视化调试快速部署工具
  • 原文地址:https://www.cnblogs.com/yjbjingcha/p/6928936.html
Copyright © 2011-2022 走看看