zoukankan      html  css  js  c++  java
  • WebSocket

    一、WebSocket简介

    • 顾名思义即:Web + Socket,它是一中TCP协议,当客户端和服务器完成握手,建立连接后,ws 就如普通 socket 一样,在两者之间进行通信。
    • WebSocket 是HTML5的新功能,在HTML5之前,HTML还不支持WebSocket,当时如果要进行实时的内容更新则需要使用到轮询技术。

    轮询

    客户端一直发请求,服务端一直收请求,直到服务端给客户端响应。

    • 优点:响应及时

    • 缺点:消耗两端资源,非常严重,占用大量带宽

    长轮询

    客户端发起请求,服务收到请求,内部进行轮询等待

    • 优点:响应及时,节省资源,不会一直发送求情和响应

    • 缺点:服务器资源消耗

    二、如何使用WebSocket

    1. Flask+WebSocket(基础版low)

    from flask import Flask,request,render_template
    from geventwebsocket.websocket import WebSocket  # Websocket语法提示
    from geventwebsocket.handler import WebSocketHandler  # 处理Websocket请求头
    from gevent.pywsgi import WSGIServer  # 在处理http请求时还能处理websocket请求
    
    app = Flask(__name__)
    
    @app.route("/index")
    def index():
    
        return render_template("ws.html")
    
    
    @app.route("/web")
    def web():
        # 获取用户的socket连接, 并开启语法提示
        user_socket = request.environ.get("wsgi.websocket")  # type:WebSocket
        if not user_socket:
            return "请使用websocket方式连接"
        while True:
            # 获取用户发送的信息
            user_msg = user_socket.receive()
            print(user_msg)
            data = input("回复:").strip()
            user_socket.send(data)
    
    
    if __name__ == '__main__':
        # app.run("0.0.0.0", 8000, debug=True)
        # 当请求进来之后,如果是websocket请求交给WebSocketHandler处理之后再交给APP处理。
        http_serv = WSGIServer(("0.0.0.0",8000), app,handler_class=WebSocketHandler)
        http_serv.serve_forever()
    Python代码
    <script type="application/javascript">
        var ws = new WebSocket("ws://127.0.0.1:8000/web");
        ws.onopen = function () {
            ws.send("加油")
        };
    
        ws.onmessage = function (serv_msg) {
            console.log(serv_msg.data)
        }
    </script>
    HTML代码

    2. Flask+WebSocket(单聊带昵称low)

    from flask import Flask, request, render_template
    from geventwebsocket.websocket import WebSocket
    from geventwebsocket.websocket import WebSocketError
    from geventwebsocket.handler import WebSocketHandler
    from gevent.pywsgi import WSGIServer
    
    import json
    
    app = Flask(__name__)
    
    
    @app.route("/index")
    def index():
        return render_template("ws.html")
    
    
    user_socket_list = []
    user_socket_dict = {}
    
    
    @app.route("/ws/<username>")
    def ws(username):
        user_socket = request.environ.get("wsgi.websocket")  # type:WebSocket
        if not user_socket:
            return "请使用WebSocket方式连接"
        user_socket_dict[username] = user_socket
        print(user_socket_dict)
        while True:
            try:
                user_msg = user_socket.receive()
                # 将字符类型的字典,转换为字典类型
                user_msg =eval(user_msg)
                print()
                for user_name, u_socket in user_socket_dict.items():
                    if user_name == user_msg["to_user"]:
                        who_send_msg = {
                            "send_user": username,
                            "send_msg": user_msg['send_msg']
                        }
                        u_socket.send(json.dumps(who_send_msg))
            except WebSocketError as e:
                user_socket_dict.pop(username)
                print(e)
                return "断开连接"
    
    
    
    
    
    if __name__ == '__main__':
        # app.run("0.0.0.0",8000,debug=True)
        http_serv = WSGIServer(("0.0.0.0", 8000), app, handler_class=WebSocketHandler)
        http_serv.serve_forever()
    Python代码
    <!DOCTYPE html>
    <html lang="en">
    <head>
        <meta charset="UTF-8">
        <title>Title</title>
        <meta name="viewport" content="width=device-width, initial-scale=1">
    </head>
    <body>
    你是谁:<input type="text" id="username">
    给谁发:<input type="text" id="to_user">
    <button id="create_ws" onclick="go_to()">创建ws连接</button>
    <div style=" 400px; height: 300px;" id="chat_window">
    
    </div>
    <div>
        <input type="text" id="send_msg">
        <button id="btn_send">发送消息</button>
    </div>
    </body>
    <script type="application/javascript">
        var ws_url = "ws://127.0.0.1:8000/ws/";
        var ws = null;
        //    ws.onopen = function () {
        //        ws.send("你好!")
        //    };a
        function go_to() {
            var username = document.getElementById("username");
            ws = new WebSocket(ws_url + username.value);
    
            ws.onmessage = function (serv_msg) {
                msg = JSON.parse(serv_msg.data);
                console.log(msg);
                create_chat("y",msg);
            };
        }
    
    
        function create_chat(self, content) {
            if (self == "w") {
                self = "right";
                var spantag = document.createElement("span");
                spantag.innerText = content.send_msg;
                var spantag1 = document.createElement("span");
                spantag1.innerText = ":我";
            } else {
                self = "left";
                var spantag = document.createElement("span");
                spantag.innerText = content.send_user + ":";
                var spantag1 = document.createElement("span");
                spantag1.innerText = content.send_msg;
    
            }
    
            var divtag = document.createElement("div");
            divtag.style = "text-align:" + self;
    
            divtag.appendChild(spantag);
            divtag.appendChild(spantag1);
            var chat_window = document.getElementById("chat_window");
            chat_window.appendChild(divtag);
        }
    
        document.getElementById("btn_send").addEventListener("click", function () {
            var send_msg = document.getElementById("send_msg");
            var to_user = document.getElementById("to_user");
            send_msg_json = {
                send_msg:send_msg.value,
                to_user :to_user.value
            };
            ws.send(JSON.stringify(send_msg_json));
    
            var s_msg = {send_msg:send_msg.value};
            create_chat("w", s_msg);
            send_msg.value = "";
        })
    
    </script>
    </html>
    HTML代码

    3.Flask+WebSocket(群聊带昵称low)

    from flask import Flask, request, render_template
    from geventwebsocket.websocket import WebSocket
    from geventwebsocket.websocket import WebSocketError
    from geventwebsocket.handler import WebSocketHandler
    from gevent.pywsgi import WSGIServer
    
    import json
    
    app = Flask(__name__)
    
    
    @app.route("/index")
    def index():
        return render_template("ws.html")
    
    
    user_socket_list = []
    user_socket_dict = {}
    
    
    @app.route("/ws/<username>")
    def ws(username):
        user_socket = request.environ.get("wsgi.websocket")  # type:WebSocket
        if not user_socket:
            return "请使用WebSocket方式连接"
        user_socket_dict[username] = user_socket
        print(user_socket_dict)
        while True:
            try:
                user_msg = user_socket.receive()
                for user_name, u_socket in user_socket_dict.items():
                    who_send_msg = {
                        "send_user": username,
                        "send_msg": user_msg
                    }
                    if user_socket == u_socket:
                        continue
                    u_socket.send(json.dumps(who_send_msg))
            except WebSocketError as e:
                user_socket_dict.pop(username)
                print(user_socket_dict)
                print(e)
    
    
    if __name__ == '__main__':
        # app.run("0.0.0.0",9527,debug=True)
        http_serv = WSGIServer(("0.0.0.0", 8000), app, handler_class=WebSocketHandler)
        http_serv.serve_forever()
    Python代码
    <!DOCTYPE html>
    <html lang="en">
    <head>
        <meta charset="UTF-8">
        <title>Title</title>
        <meta name="viewport" content="width=device-width, initial-scale=1">
    </head>
    <body>
    你是谁:<input type="text" id="username">
    <button id="create_ws" onclick="go_to()">创建ws连接</button>
    <div style=" 400px; height: 300px;" id="chat_window">
    
    </div>
    <div>
        <input type="text" id="send_msg">
        <button id="btn_send">发送消息</button>
    </div>
    </body>
    <script type="application/javascript">
        var ws_url = "ws://127.0.0.1:8000/ws/";
        var ws = null;
    
        function go_to() {
            var username = document.getElementById("username");
            ws = new WebSocket(ws_url + username.value);
    
            ws.onmessage = function (serv_msg) {
                msg = JSON.parse(serv_msg.data);
                create_chat("y",msg);
            };
        }
    
        function create_chat(self, content) {
            if (self == "w") {
                self = "right";
                var spantag = document.createElement("span");
                spantag.innerText = content.send_msg;
                var spantag1 = document.createElement("span");
                spantag1.innerText = ":我";
            } else {
                self = "left";
                var spantag = document.createElement("span");
                spantag.innerText = content.send_user + ":";
                var spantag1 = document.createElement("span");
                spantag1.innerText = content.send_msg;
            }
    
            var divtag = document.createElement("div");
            divtag.style = "text-align:" + self;
    
            divtag.appendChild(spantag);
            divtag.appendChild(spantag1);
            var chat_window = document.getElementById("chat_window");
            chat_window.appendChild(divtag);
        }
    
        document.getElementById("btn_send").addEventListener("click", function () {
            var send_msg = document.getElementById("send_msg");
            ws.send(send_msg.value);
            var s_msg = {send_msg:send_msg.value};
            create_chat("w", s_msg);
            send_msg.value = "";
        })
    
    </script>
    </html>
    HTML代码

    三、WebSocket + Flask实现实时更新

    实现思路:

    1. 客户端发起 ws 连接请求。
    2. 服务器响应,并以key、vlue形式存储用户与用户地址。

    3. 当某一 ws 向服务器发送信息时,服务器遍历字典中,向里面每一个用户发送信息。

    4. ws 断开连接时,从用户字典中剔除用户信息。

    创建服务端

  • 相关阅读:
    vim中字符串的替换
    利用nginx 来实现内网yum源(反向代理)
    linux下分卷压缩,合并解压的3种方法
    Lua io.lines()
    Springboot入门:
    GitHub起步---创建第一个项目
    常用的git命令
    Linux下安装Git
    Mysql连接报错:1130-host ... is not allowed to connect to this MySql server如何处理
    windows下开启远程连接Mysql
  • 原文地址:https://www.cnblogs.com/Dream-huang/p/9513817.html
Copyright © 2011-2022 走看看