Websocket
即时通讯1.需求 即时通讯工具一定要保障的是即时性
基于现在的通讯协议HTTP
要如何保障即时性呢?
2.短连接型 基于HTTP
短连接如何保障数据的即时性
HTTP
的特性就是无状态的短连接,即一次请求一次响应断开连接失忆,这样服务端就无法主动的去寻找客户端给客户端主动推送消息
1.轮询
即:客户端不断向服务器发起请求索取消息
优点:基本保障消息即时性
缺点:大量的请求导致客户端和服务端的压力倍增
2.长轮询
即:客户端向服务器发起请求,在HTTP
最大超时时间内不断开请求获取消息,超时后重新发起请求
优点:基本保障消息即时性
缺点:长期占用客户端独立线程,长期占用服务端独立线程,服务器压力倍增
3.长连接型 基于socket
长连接,由于长连接是双向且有状态的保持连接,所以服务端可以有效的主动的向客户端推送数据
1.socketio
长连接协议
优点:消息即时,兼容性强
缺点:接入复杂度高,为保障兼容性冗余依赖过多较重
2.websocket
长连接协议
优点:消息即时,轻量级,灵活适应多场景,机制更加成熟
缺点:相比socket
兼容性较差
总体来说,Socketio
紧紧只是为了解决通讯而存在的,而Websocket
是为了解决更多更复杂的场景通讯而存在的
这里推荐Websocket
的原因是因为,我们的Django
框架甚至是Flask
框架,都有成熟的第三方库
而且Tornado
框架集成Websocket
4.Django
实现Websocket
使用Django
来实现Websocket
服务的方法很多在这里我们推荐技术最新的Channels
库来实现
4.1.安装DjangoChannels
Channels
安装如果你是Windows
操作系统的话,那么必要条件就是Python3.7
4.2.配置DjangoChannels
1.创建项目ChannelsReady
1 django-admin startprobject ChannelsReady
2.在项目的settings.py
同级目录中,新建文件routing.py
1 2 3 4 5 6 from channels.routing import ProtocolTypeRouterapplication = ProtocolTypeRouter({ })
3.在项目配置文件settings.py
中写入
1 2 3 4 5 INSTALLED_APPS = [ 'channels' ] ASGI_APPLICATION = "ChannelsReady.routing.application"
4.3.启动带有Channels
提供的ASGI
的Django
项目 1 2 3 4 5 6 You have 17 unapplied migration(s). Your project may not work properly until you apply the migrations for app(s): admin, auth, contenttypes, sessions. Run 'python manage.py migrate' to apply them. February 01, 2020 - 17:27:13 Django version 3.0.2, using settings 'ChannelsReady.settings' Starting ASGI/Channels version 2.4.0 development server at http://0.0.0.0:8000/ Quit the server with CTRL-BREAK.
很明显可以看到ASGI/Channels
,这样就算启动完成了
4.4.创建Websocket
服务 1.创建一个新的应用chats
1 python manage.py startapp chats
2.在settings.py
中注册chats
1 2 3 4 INSTALLED_APPS = [ 'chats' , 'channels' ]
3.在chats
应用中新建文件chatService.py
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 from channels.generic.websocket import WebsocketConsumerclass ChatService (WebsocketConsumer) : def connect (self) : pass def receive (self, text_data=None, bytes_data=None) : pass def disconnect (self, code) : pass
4.5.为Websocket
处理对象增加路由 1.在chats
应用中,新建urls.py
1 2 3 4 5 from django.urls import pathfrom chats.chatService import ChatServicewebsocket_url = [ path("ws/" ,ChatService) ]
2.回到项目routing.py
文件中增加ASGI
非HTTP
请求处理
1 2 3 4 5 6 7 8 from channels.routing import ProtocolTypeRouter,URLRouterfrom chats.urls import websocket_urlapplication = ProtocolTypeRouter({ "websocket" :URLRouter( websocket_url ) })
总结:
下载
注册到setting.py里的app
在setting.py同级的目录下注册channels使用的路由—–>routing.py
将routing.py注册到setting.py
把urls.py的路由注册到routing.py里
编写wsserver.py来处理websocket请求
5.websocket客户端 5.1.基于vue的websocket客户端 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 <template> <div> <input type="text" v-model="message"> <p><input type="button" @click="send" value="发送"></p> <p><input type="button" @click="close_socket" value="关闭"></p> </div> </template> <script> export default { name:'websocket1', data() { return { message:'', testsocket:'' } }, methods:{ send(){ // send 发送信息 // close 关闭连接 this.testsocket.send(this.message) this.testsocket.onmessage = (res) => { console.log("WS的返回结果",res.data); } }, close_socket(){ this.testsocket.close() } }, mounted(){ this.testsocket = new WebSocket("ws://127.0.0.1:8000/ws/") // onopen 定义打开时的函数 // onclose 定义关闭时的函数 // onmessage 定义接收数据时候的函数 // this.testsocket.onopen = function(){ // console.log("开始连接socket") // }, // this.testsocket.onclose = function(){ // console.log("socket连接已经关闭") // } } } </script>
6.广播消息 6.1客户端保持不变,同时打开多个客户端 6.2服务端存储每个链接的对象 1 2 3 4 5 6 7 8 9 10 11 12 13 14 socket_list = [] class ChatService (WebsocketConsumer) : def connect (self) : self.accept() socket_list.append(self) def receive (self, text_data=None, bytes_data=None) : print(text_data) for ws in socket_list: ws.send(text_data)
7.点对点消息 7.1客户端将用户名拼接到url,并在发送的消息里指明要发送的对象 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 <template> <div> <input type="text" v-model="message"> <input type="text" v-model="user"> <p><input type="button" @click="send" value="发送"></p> <p><input type="button" @click="close_socket" value="关闭"></p> </div> </template> <script> export default { name:'websocket1', data() { return { message:'', testsocket:'', user:'' } }, methods:{ send(){ // send 发送信息 // close 关闭连接 var data1 = {"message":this.message,"to_user":this.user} this.testsocket.send(JSON.stringify(data1)) this.testsocket.onmessage = (res) => { console.log("WS的返回结果",res.data); } }, close_socket(){ this.testsocket.close() }, generate_uuid: function() { var d = new Date().getTime(); if (window.performance && typeof window.performance.now === "function") { d += performance.now(); //use high-precision timer if available } var uuid = "xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx".replace( /[xy]/g, function(c) { var r = (d + Math.random() * 16) % 16 | 0; d = Math.floor(d / 16); return (c == "x" ? r : (r & 0x3) | 0x8).toString(16); } ); return uuid; }, }, mounted(){ var username = this.generate_uuid(); console.log(username) this.testsocket = new WebSocket("ws://127.0.0.1:8000/ws/"+ username +"/") console.log(this.testsocket) this.testsocket.onmessage = (res) => { console.log("WS的返回结果",res.data); } // onopen 定义打开时的函数 // onclose 定义关闭时的函数 // onmessage 定义接收数据时候的函数 // this.testsocket.onopen = function(){ // console.log("开始连接socket") // }, // this.testsocket.onclose = function(){ // console.log("socket连接已经关闭") // } } } </script>
7.2服务端存储用户名以及websocketConsumer,然后给对应的用户发送信息 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 from channels.generic.websocket import WebsocketConsumeruser_dict ={} list = [] import jsonclass ChatService (WebsocketConsumer) : def connect (self) : self.accept() username = self.scope.get("url_route" ).get("kwargs" ).get("username" ) user_dict[username] =self print(user_dict) def receive (self, text_data=None, bytes_data=None) : data = json.loads(text_data) print(data) to_user = data.get("to_user" ) message = data.get("message" ) ws = user_dict.get(to_user) print(to_user) print(message) print(ws) ws.send(text_data) def disconnect (self, code) : pass