1.群聊
- web - socket--基于TCP/UDP
- http - 无状态的短链接
- 长连接:客户端和服务器保持永久性的链接,除非有一方主动断开,
- 轮询:客户端和服务端不断连接,然后断开,请求响应;不能保证数据的实时性.
- 长轮询:长轮询:客户端发起请求至server,服务端不响应,服务端一直等待,链接一直建立,等待http链接自动超时(默认15s),主动断开链接
1.1 服务端
from flask import Flask, request, render_template
from geventwebsocket.handler import WebSocketHandler
from geventwebsocket.server import WSGIServer
from geventwebsocket.websocket import WebSocket
app = Flask(__name__)
socket_lsit = []
@app.route('/ws')
def my_ws():
ws_socket = request.environ.get('wsgi.websocket')
socket_lsit.append(ws_socket)
print(len(socket_lsit))
while True:
msg = ws_socket.receive()
print(msg)
for usocket in socket_lsit:
if usocket == ws_socket:
continue
try:
usocket.send(msg)
except:
continue
@app.route('/wechat')
def wechat():
return render_template('ws_we.html')
if __name__ == '__main__':
http_serv = WSGIServer(('0.0.0.0', 9527),
app,
handler_class=WebSocketHandler
)
http_serv.serve_forever()
1.2 客户端(html文件)
- 状态码status:
- 1 当前连接处于可用状态
- 3 由服务器主动发起断开
- 0 正在建立连接或连接建立失败
- 2 客户端主动发起断开
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta http-equiv="x-ua-compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1">
<title>Title</title>
</head>
<body>
<input type="text" id="content"><button onclick="send_msg()">发送消息</button>
<div id="content_list">
</div>
</body>
<script type="application/javascript">
var ws = new WebSocket("ws://192.168.12.10:9527/ws");
ws.onmessage = function (messageEvent) {
console.log(messageEvent.data);
var my_div = document.getElementById("content_list");
var ptag = document.createElement("p");
ptag.innerText = messageEvent.data;
my_div.appendChild(ptag);
};
function send_msg() {
var msg = document.getElementById("content").value;
ws.send(msg);
}
</script>
</html>
2.单聊
2.1服务端
import json
from flask import Flask, request, render_template
from geventwebsocket.handler import WebSocketHandler
from geventwebsocket.server import WSGIServer
from geventwebsocket.websocket import WebSocket
app = Flask(__name__)
socket_dict = {}
@app.route('/ws/<username>')
def my_ws(username):
ws_socket = request.environ.get('wsgi.websocket')
print(ws_socket, username)
socket_dict[username] = ws_socket
print(len(socket_dict), socket_dict)
while True:
msg = ws_socket.receive()
msg_dict = json.loads(msg)
receiver = msg_dict.get('receiver')
receiver_socket = socket_dict.get(receiver)
receiver_socket.send(msg)
@app.route('/wechat')
def wechat():
return render_template('ws_one.html')
if __name__ == '__main__':
http_serv = WSGIServer(('0.0.0.0', 9527), app, handler_class=WebSocketHandler)
http_serv.serve_forever()
2.2客户端(html文件)
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Title</title>
</head>
基于JavaScript 实现Websocket客户端
<body>
<p>你的昵称<input type="text" id="username">
<button onclick="login()">登录聊天室</button>
</p>
<p>给<input type="text" id="receiver">发送</p>
<input type="text" id="content">
<button onclick="send_msg()">发送消息</button>
<div id="content_list" style="width: 300px">
</div>
<script type="application/javascript">
var ws = null;
function send_msg() {
var msg = document.getElementById('content').value;
var receiver = document.getElementById('receiver').value;
var sender = document.getElementById('username').value;
var send_str = {
receiver: receiver,
sender: sender,
data: msg,
};
ws.send(JSON.stringify(send_str));
var my_div = document.getElementById('content_list');
var ptag = document.createElement('p');
ptag.innerText = msg + " : " + '我';
ptag.style.cssText = 'text-align:right';
my_div.appendChild(ptag);
}
function login() {
var username = document.getElementById('username').value;
ws = new WebSocket('ws://192.168.12.10:9527/ws/' + username);
ws.onmessage = function (messageEvent) {
console.log(messageEvent.data);
var obj = JSON.parse(messageEvent.data);
var my_div = document.getElementById('content_list');
var ptag = document.createElement('p');
ptag.innerText = obj.sender + " : " + obj.data;
my_div.appendChild(ptag);
};
}
</script>
</body>
</html>
3. websocket 握手原理
import socket, base64, hashlib
sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
sock.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
sock.bind(('127.0.0.1', 9527))
sock.listen(5)
conn, address = sock.accept()
data = conn.recv(1024)
print(data)
"""
b'GET /ws HTTP/1.1\r\n
Host: 127.0.0.1:9527\r\n
User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:62.0) Gecko/20100101 Firefox/62.0\r\n
Accept: text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8\r\n
Accept-Language: zh-CN,zh;q=0.8,zh-TW;q=0.7,zh-HK;q=0.5,en-US;q=0.3,en;q=0.2\r\n
Accept-Encoding: gzip, deflate\r\n
Sec-WebSocket-Version: 13\r\n
Origin: http://localhost:63342\r\n
Sec-WebSocket-Extensions: permessage-deflate\r\n
Sec-WebSocket-Key: jocLOLLq1BQWp0aZgEWL5A==\r\n
Cookie: session=6f2bab18-2dc4-426a-8f06-de22909b967b\r\n
Connection: keep-alive, Upgrade\r\n
Pragma: no-cache\r\n
Cache-Control: no-cache\r\n
Upgrade: websocket\r\n\r\n'
"""
magic_string = '258EAFA5-E914-47DA-95CA-C5AB0DC85B11'
def get_headers(data):
header_dict = {}
header_str = data.decode("utf8")
for i in header_str.split("\r\n"):
if str(i).startswith("Sec-WebSocket-Key"):
header_dict["Sec-WebSocket-Key"] = i.split(":")[1].strip()
return header_dict
def get_header(data):
"""
将请求头格式化成字典
:param data:
:return:
"""
header_dict = {}
data = str(data, encoding='utf-8')
header, body = data.split('\r\n\r\n', 1)
header_list = header.split('\r\n')
for i in range(0, len(header_list)):
if i == 0:
if len(header_list[i].split(' ')) == 3:
header_dict['method'], header_dict['url'], header_dict['protocol'] = header_list[i].split(' ')
else:
k, v = header_list[i].split(':', 1)
header_dict[k] = v.strip()
return header_dict
headers = get_headers(data)
response_tpl = "HTTP/1.1 101 Switching Protocols\r\n" \
"Upgrade:websocket\r\n" \
"Connection: Upgrade\r\n" \
"Sec-WebSocket-Accept: %s\r\n" \
"WebSocket-Location: ws://127.0.0.1:9527\r\n\r\n"
value = headers['Sec-WebSocket-Key'] + magic_string
print(value,"magic+websocketkey")
ac = base64.b64encode(hashlib.sha1(value.encode('utf-8')).digest())
print(ac,"accept")
response_str = response_tpl % (ac.decode('utf-8'))
conn.send(response_str.encode("utf8"))
while True:
msg = conn.recv(8096)
print(msg)
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
· 基于Microsoft.Extensions.AI核心库实现RAG应用
· Linux系列:如何用heaptrack跟踪.NET程序的非托管内存泄露
· TypeScript + Deepseek 打造卜卦网站:技术与玄学的结合
· 阿里巴巴 QwQ-32B真的超越了 DeepSeek R-1吗?
· 【译】Visual Studio 中新的强大生产力特性
· 【设计模式】告别冗长if-else语句:使用策略模式优化代码结构
· AI与.NET技术实操系列(六):基于图像分类模型对图像进行分类