flask 第五章 WebSocket GeventWebsocket 单聊群聊 握手 解密 加密

1.WebSocket

首先我们来回顾一下,我们之前用socket学习过的项目有:

  1.django

  2.flask

  3.FTP - 文件服务

HTTP - TCP (特点):

  1.一次请求,一次响应,然后断开

  2.客户端永远处于主动状态

  3.服务端永远处于被动状态

  4.HTTP请求是无状态的 -- 在服务器不保存客户端的信息

  5.由于HTTP请求是无状态的,所以服务器无法主动找到客户端

   优点 : 

    1.速度快 

    2.信息安全

    3.不是特别占用资源

1.轮询:

  客户端不停的向服务端发送请求,服务端不停的向客户端响应,直到客户端拿到数据为止.

  劣势:

    1.服务端和客户端双端资源浪费

    2.带宽资源占用(带宽 : 网络连接时所能传送的最大数据流速)

    3.不能保证数据的实时性

2. 长轮询:

  客户端向服务端发送一个请求,服务端保持这个请求,不返回值就不响应,一定时间后,服务器抛弃这个请求或者是返回. 客户端收到请求后,立即再次发起保持连接.

在这里我们类比一个实例:

  你去传达室问大爷有没有快递,大爷款待你喝茶(喝茶的时候就相当于保持连接),如果中途你想去厕所(去厕所就相当于是断开了连接),回来之后继续喝茶(再次建立保持连接).

  劣势:

    1.相比于轮询方式,长轮询方式只是浪费了服务器的资源

    2.无法保证是数据实时性

  优势:

    1.相比于轮询方式,他节省了客户端的资源

    2.保证数据有效

3.长连接

  永久的保持连接(现在我们用的基本上都是长连接)

  劣势:

      对服务器的CPU要求比较高

  优势:

    1.节省了大量的资源(客户端和服务端的资源都节省了)

    2.数据实时有效性

    3.带宽资源几乎不占用

WebSocket协议 : ws协议 ws://127.0.0.1 用JavaScript封装客户端

  三方 : geventWebsocket Flask

  web框架 : 用来进行Http请求的处理(握手)

4.单聊 

.py文件的代码如下:

import json

from flask import Flask, render_template, request
from geventwebsocket.handler import WebSocketHandler
from gevent.pywsgi import WSGIServer
from geventwebsocket.websocket import WebSocket
from geventwebsocket.exceptions import WebSocketError

app = Flask(__name__)

user_socket_dict = {}


@app.route('/my_app')
def my_app():
    return render_template('myapp.html')


@app.route('/my_ws/<username>')
def my_ws(username):
    user_socket = request.environ.get( 'wsgi.websocket')  
    # 'wsgi.websocket': <geventwebsocket.websocket.WebSocket object at 0x00000206FCDC1528>,
    user_socket_dict[username] = user_socket  # type:WebSocket
    # print(user_socket_dict)
    while 1:
        try:
            # 阻塞等待消息数据
            msg = user_socket.receive()
            # 将接收到的消息反序列化成字典
            msg_dict = json.loads(msg)
            # print(msg)
            to_user = msg_dict.get('to_user')
            to_user_socket = user_socket_dict.get(to_user)
            to_user_socket.send(msg)
        except WebSocketError:
            user_socket_dict.pop(username)
            return '有个地方错了'


if __name__ == '__main__':
    http_server = WSGIServer(('0.0.0.0', 9527), app, handler_class=WebSocketHandler)
    http_server.serve_forever()

myapp.html 文件代码如下:

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
</head>
<body>
你的名字:<input type="text" id="nkname">
<button onclick="open_chat()">登录聊天室</button>
<p>发送至<input type="text" id="to_user"></p>
消息<input type="text" id="message">
<button onclick="send_msg()">发送</button>
<div id="chat_list">

</div>
</body>
<script type="application/javascript">
    var ws = null;

    function open_chat() {
        var nkname = document.getElementById('nkname').value;
        ws = new WebSocket("ws://127.0.0.1:9527/my_ws/" + nkname);
        ws.onopen = function () {
            alert('欢迎' + nkname + '登录对骂平台!!')
        };
        ws.onmessage = function (EventMessage) {
            var chat = JSON.parse(EventMessage.data);
            var p = document.createElement('p');
            p.style.cssText = 'width : 250px; text-align:left';
            p.innerText = chat.from_user + '-->' + chat.message;
            document.getElementById('chat_list').appendChild(p);
        };
        ws.onclose = function () {
            console.log('断开连接了,啥情况啊? 搞事情啊');
        };
    }

    function send_msg() {
        var message = document.getElementById('message').value;
        var from_user = document.getElementById('nkname').value;
        var to_user = document.getElementById('to_user').value;
        var send_str = {
            message: message,
            from_user: from_user,
            to_user: to_user
        };
        ws.send(JSON.stringify(send_str));
        var p = document.createElement('p');
        p.style.cssText = "width: 250px;text-align: right";
        p.innerText = send_str.message + "<-我";
        document.getElementById('chat_list').appendChild(p);
    }

</script>
</html>

5.群聊

.py文件代码如下: 

from geventwebsocket.handler import WebSocketHandler
from gevent.pywsgi import WSGIServer
from geventwebsocket.websocket import WebSocket
from geventwebsocket.exceptions import WebSocketError

from flask import Flask,render_template,request

app = Flask(__name__)

user_socket_list = []

@app.route("/my_app")
def my_app():
    return render_template("my_app.html")

@app.route("/my_ws")
def my_ws():
    user_socket = request.environ.get("wsgi.websocket") 
    # type:WebSocket
    user_socket_list.append(user_socket)
    # print(len(user_socket_list),user_socket_list)
    while 1:
        try:
            msg = user_socket.receive() # 阻塞等待消息数据
        except WebSocketError:
            user_socket_list.remove(user_socket)
            return "good bye"
        for u in user_socket_list:
            if u == user_socket:
                continue
            try:
                u.send(msg)
            except :
                continue

if __name__ == '__main__':
    # app.run()
    http_serv = WSGIServer(("0.0.0.0",9527),app,handler_class=WebSocketHandler)
    http_serv.serve_forever()

myapp.html文件代码如下:

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
</head>
<body>
    <input type="text" id="message"><button onclick="send_message()">发送</button>
    <div id="chat_list">

    </div>
</body>
<script type="application/javascript">
    var ws = new WebSocket("ws://192.168.14.200:9527/my_ws");
    ws.onmessage = function (eventMessage) {
        // document.getElementById("chat_list").innerHTML += "<p>" + eventMessage.data + "</p>";
        var p = document.createElement("p");
        p.innerText = eventMessage.data;
        document.getElementById("chat_list").appendChild(p);
    };

    function send_message() {
        var message = document.getElementById("message").value;
        ws.send(message);
    }

</script>
</html>

 

posted @ 2019-04-16 20:52  想扣篮的小矮子  阅读(377)  评论(0编辑  收藏  举报