Django 實(shí)現(xiàn) Websocket 廣播、點(diǎn)對(duì)點(diǎn)發(fā)送消息的代碼
1.Django實(shí)現(xiàn)Websocket
使用Django來(lái)實(shí)現(xiàn)Websocket服務(wù)的方法很多在這里我們推薦技術(shù)最新的Channels庫(kù)來(lái)實(shí)現(xiàn)
1.1.安裝DjangoChannels
Channels安裝如果你是Windows操作系統(tǒng)的話(huà),那么必要條件就是Python3.7
pip install channels
1.2.配置DjangoChannels
1.創(chuàng)建項(xiàng)目ChannelsReady
django-admin startprobject ChannelsReady
2.在項(xiàng)目的settings.py同級(jí)目錄中,新建文件routing.py
# routing.py
from channels.routing import ProtocolTypeRouter
application = ProtocolTypeRouter({
# 暫時(shí)為空
})
3.在項(xiàng)目配置文件settings.py中寫(xiě)入
INSTALLED_APPS = [ 'channels' ] ASGI_APPLICATION = "ChannelsReady.routing.application"
1.3.啟動(dòng)帶有Channels提供的ASGI的Django項(xiàng)目
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,這樣就算啟動(dòng)完成了
1.4.創(chuàng)建Websocket服務(wù)
1.創(chuàng)建一個(gè)新的應(yīng)用chats
python manage.py startapp chats
2.在settings.py中注冊(cè)chats
INSTALLED_APPS = [ 'chats', 'channels' ]
3.在chats應(yīng)用中新建文件chatService.py
from channels.generic.websocket import WebsocketConsumer # 這里除了 WebsocketConsumer 之外還有 # JsonWebsocketConsumer # AsyncWebsocketConsumer # AsyncJsonWebsocketConsumer # WebsocketConsumer 與 JsonWebsocketConsumer 就是多了一個(gè)可以自動(dòng)處理JSON的方法 # AsyncWebsocketConsumer 與 AsyncJsonWebsocketConsumer 也是多了一個(gè)JSON的方法 # AsyncWebsocketConsumer 與 WebsocketConsumer 才是重點(diǎn) # 看名稱(chēng)似乎理解并不難 Async 無(wú)非就是異步帶有 async / await # 是的理解并沒(méi)有錯(cuò),但對(duì)與我們來(lái)說(shuō)他們唯一不一樣的地方,可能就是名字的長(zhǎng)短了,用法是一模一樣的 # 最夸張的是,基類(lèi)是同一個(gè),而且這個(gè)基類(lèi)的方法也是Async異步的 class ChatService(WebsocketConsumer): # 當(dāng)Websocket創(chuàng)建連接時(shí) def connect(self): pass # 當(dāng)Websocket接收到消息時(shí) def receive(self, text_data=None, bytes_data=None): pass # 當(dāng)Websocket發(fā)生斷開(kāi)連接時(shí) def disconnect(self, code): pass
1.5.為Websocket處理對(duì)象增加路由
1.在chats應(yīng)用中,新建urls.py
from django.urls import path
from chats.chatService import ChatService
websocket_url = [
path("ws/",ChatService)
]
2.回到項(xiàng)目routing.py文件中增加ASGI非HTTP請(qǐng)求處理
from channels.routing import ProtocolTypeRouter,URLRouter
from chats.urls import websocket_url
application = ProtocolTypeRouter({
"websocket":URLRouter(
websocket_url
)
})
總結(jié):
- 下載
- 注冊(cè)到setting.py里的app
- 在setting.py同級(jí)的目錄下注冊(cè)channels使用的路由----->routing.py
- 將routing.py注冊(cè)到setting.py
- 把urls.py的路由注冊(cè)到routing.py里
- 編寫(xiě)wsserver.py來(lái)處理websocket請(qǐng)求
<template>
<div>
<input type="text" v-model="message">
<p><input type="button" @click="send" value="發(fā)送"></p>
<p><input type="button" @click="close_socket" value="關(guān)閉"></p>
</div>
</template>
<script>
export default {
name:'websocket1',
data() {
return {
message:'',
testsocket:''
}
},
methods:{
send(){
// send 發(fā)送信息
// close 關(guān)閉連接
this.testsocket.send(this.message)
this.testsocket.onmessage = (res) => {
console.log("WS的返回結(jié)果",res.data);
}
},
close_socket(){
this.testsocket.close()
}
},
mounted(){
this.testsocket = new WebSocket("ws://127.0.0.1:8000/ws/")
// onopen 定義打開(kāi)時(shí)的函數(shù)
// onclose 定義關(guān)閉時(shí)的函數(shù)
// onmessage 定義接收數(shù)據(jù)時(shí)候的函數(shù)
// this.testsocket.onopen = function(){
// console.log("開(kāi)始連接socket")
// },
// this.testsocket.onclose = function(){
// console.log("socket連接已經(jīng)關(guān)閉")
// }
}
}
</script>
3.廣播消息
3.1客戶(hù)端保持不變,同時(shí)打開(kāi)多個(gè)客戶(hù)端
3.2服務(wù)端存儲(chǔ)每個(gè)鏈接的對(duì)象
socket_list = [] class ChatService(WebsocketConsumer): # 當(dāng)Websocket創(chuàng)建連接時(shí) def connect(self): self.accept() socket_list.append(self) # 當(dāng)Websocket接收到消息時(shí) def receive(self, text_data=None, bytes_data=None): print(text_data) # 打印收到的數(shù)據(jù) for ws in socket_list: # 遍歷所有的WebsocketConsumer對(duì)象 ws.send(text_data) # 對(duì)每一個(gè)WebsocketConsumer對(duì)象發(fā)送數(shù)據(jù)
4.點(diǎn)對(duì)點(diǎn)消息
4.1客戶(hù)端將用戶(hù)名拼接到url,并在發(fā)送的消息里指明要發(fā)送的對(duì)象
<template>
<div>
<input type="text" v-model="message">
<input type="text" v-model="user">
<p><input type="button" @click="send" value="發(fā)送"></p>
<p><input type="button" @click="close_socket" value="關(guān)閉"></p>
</div>
</template>
<script>
export default {
name:'websocket1',
data() {
return {
message:'',
testsocket:'',
user:''
}
},
methods:{
send(){
// send 發(fā)送信息
// close 關(guān)閉連接
var data1 = {"message":this.message,"to_user":this.user}
this.testsocket.send(JSON.stringify(data1))
this.testsocket.onmessage = (res) => {
console.log("WS的返回結(jié)果",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的返回結(jié)果",res.data);
}
// onopen 定義打開(kāi)時(shí)的函數(shù)
// onclose 定義關(guān)閉時(shí)的函數(shù)
// onmessage 定義接收數(shù)據(jù)時(shí)候的函數(shù)
// this.testsocket.onopen = function(){
// console.log("開(kāi)始連接socket")
// },
// this.testsocket.onclose = function(){
// console.log("socket連接已經(jīng)關(guān)閉")
// }
}
}
</script>
4.2服務(wù)端存儲(chǔ)用戶(hù)名以及websocketConsumer,然后給對(duì)應(yīng)的用戶(hù)發(fā)送信息
from channels.generic.websocket import WebsocketConsumer
user_dict ={}
list = []
import json
class ChatService(WebsocketConsumer):
# 當(dāng)Websocket創(chuàng)建連接時(shí)
def connect(self):
self.accept()
username = self.scope.get("url_route").get("kwargs").get("username")
user_dict[username] =self
print(user_dict)
# list.append(self)
# 當(dāng)Websocket接收到消息時(shí)
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)
# 當(dāng)Websocket發(fā)生斷開(kāi)連接時(shí)
def disconnect(self, code):
pass
總結(jié)
到此這篇關(guān)于Django 實(shí)現(xiàn) Websocket 廣播、點(diǎn)對(duì)點(diǎn)發(fā)送消息的文章就介紹到這了,更多相關(guān)Django 實(shí)現(xiàn) Websocket 廣播、點(diǎn)對(duì)點(diǎn)發(fā)送消息內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Python技巧分享之如何將字符串轉(zhuǎn)回DataFrame格式
平常我們使用pandas,一般使用的是DataFrame和Series,但個(gè)別交換數(shù)據(jù)的時(shí)候,只能使用字符串,那如何再將字符串再轉(zhuǎn)回DataFrame格式呢,本文就來(lái)和大家講講解決辦法2023-06-06
將Django使用的數(shù)據(jù)庫(kù)從MySQL遷移到PostgreSQL的教程
這篇文章主要介紹了將Django使用的數(shù)據(jù)庫(kù)從MySQL遷移到PostgreSQL的教程,同時(shí)提到了一些注意事項(xiàng),需要的朋友可以參考下2015-04-04
利用Python編寫(xiě)個(gè)冷笑話(huà)生成器
這篇文章主要為大家分享了如何利用Python編寫(xiě)一個(gè)簡(jiǎn)易的冷笑話(huà)生成器,這樣再惹女朋友生氣,就可以說(shuō)笑話(huà)哄她啦,快跟隨小編一起學(xué)習(xí)一下2022-04-04
利用python爬取古詩(shī)文網(wǎng)中各類(lèi)古詩(shī)的方法
這篇文章主要介紹了利用python爬取古詩(shī)文網(wǎng)中各類(lèi)古詩(shī)的方法,文中通過(guò)代碼示例給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作有一定的幫助,需要的朋友可以參考下2024-03-03
Python中if __name__ == ''__main__''作用解析
這篇文章主要介紹了Python中if __name__ == '__main__'作用解析,這斷代碼在Python中非常常見(jiàn),它有作用?本文就解析了它的作用,需要的朋友可以參考下2015-06-06
Pycharm中安裝Pygal并使用Pygal模擬擲骰子(推薦)
這篇文章主要介紹了Pycharm中安裝Pygal并使用Pygal模擬擲骰子,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-04-04
YOLOv5改進(jìn)系列之增加小目標(biāo)檢測(cè)層
yolov5出來(lái)已經(jīng)很長(zhǎng)時(shí)間了,所以有關(guān)yolov5的一些詳細(xì)介紹在這里就不一一介紹了,下面這篇文章主要給大家介紹了關(guān)于YOLOv5改進(jìn)系列之增加小目標(biāo)檢測(cè)層的相關(guān)資料,需要的朋友可以參考下2022-09-09

