自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

幾行代碼,擼了個 元宇宙?!

開發(fā) 前端
Facebook 改名為 meta,一下子點燃了 元宇宙 這個概念。今天我就用 Python 實現(xiàn)一個簡單的迷你元宇宙。代碼簡潔易懂,不僅可以學(xué)習(xí) Python 知識,還能用實踐理解元宇宙的概念。

[[439939]]

Facebook 改名為 meta,一下子點燃了 元宇宙 這個概念。

今天我就用 Python 實現(xiàn)一個簡單的迷你元宇宙。

代碼簡潔易懂,不僅可以學(xué)習(xí) Python 知識,還能用實踐理解元宇宙的概念。

還等什么,現(xiàn)在就開始吧!

迷你元宇宙

什么是元宇宙?

不同的人有不同的理解和認(rèn)識,最能達(dá)成共識的是:

元宇宙是個接入點,每個人都可以成為其中的一個元素,彼此互動。

那么我們的元宇宙有哪些功能呢?

首先必須有可以接入的功能。

然后彼此之間可以交流信息。比如 a 發(fā)消息給 b,b 可以發(fā)消息給 a,同時可以將消息廣播出去,也就是成員之間,可以私信 和 群聊。

另外,在元宇宙的成員可以收到元宇宙的動態(tài),比如新人加入,或者有人離開等,如果玩膩了,可以離開元宇宙。

最終的效果像這樣:

效果

設(shè)計

如何構(gòu)建接入點

直接思考可能比較困難,換個角度想,接入點其實就是 —— 服務(wù)器。

只要是上網(wǎng),每時每刻,我們都是同服務(wù)器打交的。

那就選擇最簡單的 TCP 服務(wù)器,TCP 服務(wù)器的核心是維護(hù)套接字(socket)的狀態(tài),向其中發(fā)送或者獲取信息。

python 的 socket 庫,提供了很多有關(guān)便捷方法,可以幫助我們構(gòu)建。

核心代碼如下:

  1. import socket 
  2.  
  3. socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM) 
  4. socket.bind((ip, port)) 
  5. socket.listen() 
  6.  
  7. data = socket.recv(1024) 
  8. ... 

創(chuàng)建一個 socket,讓其監(jiān)聽機(jī)器所擁有的一個 ip 和 端口,然后從 socket 中讀取發(fā)送過來的數(shù)據(jù)。

如何構(gòu)建客戶端

客戶端是為了方便用戶鏈接到元宇宙的工具,這里,就是能鏈接到服務(wù)器的工具,服務(wù)器是 TCP 服務(wù)器,客戶端自然需要用可以鏈接 TCP 服務(wù)器的方式。

python 也已為我們備好,幾行代碼就可以搞定,核心代碼如下:

  1. import socket 
  2.  
  3. client = socket.socket(socket.AF_INET,socket.SOCK_STREAM) 
  4. client.connect((ip, port)) 
  5.  
  6. data = client.recv(1024) 
  7. ... 

代碼與服務(wù)器很像,不過去鏈接一個服務(wù)器的 ip 和 端口。

如何構(gòu)建業(yè)務(wù)邏輯

首先需要讓服務(wù)器將接入的用戶管理起來。

然后當(dāng)接收到用戶消息時做出判斷,是轉(zhuǎn)發(fā)給其他用戶,廣播還是做出回應(yīng)。

這樣就需要構(gòu)造一種消息格式,用來表示用戶消息的類型或者目的。

我們就用 @username 的格式來區(qū)分,消息發(fā)給特殊用戶還是群發(fā)。

另外,為了完成注冊功能,需要再定義一種命令格式,用于設(shè)置 username,我們可以用 name:username 的格式作為設(shè)置用戶名的命令。

構(gòu)建

有了初步設(shè)計,就可以進(jìn)一步構(gòu)建我們的代碼了。

服務(wù)端

服務(wù)器需要同時響應(yīng)多個鏈接,其中包括新鏈接創(chuàng)建,消息 和 鏈接斷開 等。

為了不讓服務(wù)器阻塞,我們采用非阻塞的鏈接,當(dāng)鏈接接入時,將鏈接存儲起來,然后用 select 工具,等待有了消息的鏈接。

這個功能,已經(jīng)有人實現(xiàn)好了 simpletcp[1],只要稍作改動就好。

將其中的收到消息,建立鏈接,關(guān)閉鏈接做成回調(diào)方法,以便再外部編寫業(yè)務(wù)邏輯。

核心業(yè)務(wù)

這里說明一下核心代碼:

  1. # 創(chuàng)建一個服務(wù)器鏈接 
  2. self._socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM) 
  3. self._socket.setblocking(0) 
  4. self._socket.bind((self.ip, self.port)) 
  5. self._socket.listen(self._max_connections) 
  6.  
  7. # 存放已建立的鏈接 
  8. readers = [self._socket] 
  9. # 存放客戶端 ip和端口 
  10. IPs = dict() 
  11.  
  12. # 退出標(biāo)記 用于關(guān)閉服務(wù)器 
  13. self._stop = False 
  14.  
  15. # 服務(wù)器主循環(huán) 
  16. while readers and not self._stop: 
  17.     # 利用 select 從 建立的鏈接中選取一些有新消息的 
  18.     read, _, err = select.select(readers, [], readers) 
  19.      
  20.     for sock in read
  21.         if sock is self._socket: 
  22.             # 建立了新鏈接 
  23.  
  24.             # 獲取新鏈接的 socket 以及 ip和端口 
  25.             client_socket, client_ip = self._socket.accept() 
  26.              
  27.             # 將鏈接設(shè)置為非阻塞的 
  28.             client_socket.setblocking(0) 
  29.             # 添加到監(jiān)聽隊列 
  30.             readers.append(client_socket) 
  31.             # 存儲ip信息 
  32.             IPs[client_socket] = client_ip 
  33.  
  34.             # 調(diào)用建立鏈接回調(diào)函數(shù) 
  35.             self.onCreateConn(self, client_socket, client_ip) 
  36.         else
  37.             # 收到了新消息 
  38.             try: 
  39.                 # 獲取消息 
  40.                 data = sock.recv(self.recv_bytes) 
  41.             except socket.error as e: 
  42.                 if e.errno == errno.ECONNRESET: 
  43.                     # 表明鏈接已退出 
  44.                     data = None 
  45.                 else
  46.                     raise e 
  47.             if data: 
  48.                 # 調(diào)用收到消息回調(diào)函數(shù) 
  49.                 self.onReceiveMsg(self, sock, IPs[sock], data) 
  50.             else
  51.                 # 鏈接退出時,移除監(jiān)聽隊列 
  52.                 readers.remove(sock) 
  53.                 sock.close() 
  54.  
  55.                 # 調(diào)用鏈接關(guān)閉回調(diào)函數(shù) 
  56.                 self.onCloseConn(self, sock, IPs[sock])          
  57.     # 處理存在錯誤的鏈接 
  58.     for sock in err: 
  59.         # 移除監(jiān)聽隊列 
  60.         readers.remove(sock) 
  61.         sock.close() 
  62.  
  63.         # 調(diào)用鏈接關(guān)閉回調(diào)函數(shù) 
  64.         self.onCloseConn(self, sock, IPs[sock]) 
  • 首先利用 socket 建立一個服務(wù)器鏈接,這個和最初的 socket 核心代碼一樣
  • 不同的是設(shè)置鏈接為非阻塞的,這樣就可以通過 select 同時監(jiān)控多個鏈接,也不至于阻塞服務(wù)器了。關(guān)于 select 可以看這里[2]
  • 在主循環(huán)中,篩選出有了消息的鏈接,判斷是建立鏈接還是消息發(fā)送,調(diào)用不同的回調(diào)函數(shù)
  • 最后處理一下異常

事件處理

現(xiàn)在通過回調(diào)函數(shù),就可以編寫業(yè)務(wù)了,之間看代碼。

這段是建立鏈接時的處理:

  1. def onCreateConn(server, sock, ip): 
  2.  
  3. cid = f'{ip[0]}_{ip[1]}' 
  4.  
  5. clients[cid] = {'cid': cid, 'sock': sock, 'name': None} 
  6.  
  7. sock.send("你已經(jīng)接入元宇宙,告訴我你的代號,輸入格式為 name:lily.".encode('utf-8')) 
  • 首先計算出客戶端 id,即 cid,通過 ip 和 端口 組成
  • clients 是個詞典,用 cid 為 key,存儲了 cid、鏈接、和名稱
  • 一旦建立起來鏈接,向鏈接發(fā)送一段問候語,并要求其設(shè)置自己的名稱

然后是接收消息的回調(diào)函數(shù),這個相對復(fù)雜一些,主要是處理的情況更多:

  1. def onReceiveMsg(server, sock, ip, data): 
  2.     cid = f'{ip[0]}_{ip[1]}' 
  3.     data = data.decode('utf-8'
  4.     print(f"收到數(shù)據(jù): {data}"
  5.     _from = clients[cid] 
  6.     if data.startswith('name:'): 
  7.         # 設(shè)置名稱 
  8.         name = data[5:].strip() 
  9.         if not name
  10.             sock.send(f"不能設(shè)置空名稱,否則其他人找不見你".encode('utf-8')) 
  11.         elif not checkname(name, cid): 
  12.             sock.send(f"這個名字{name}已經(jīng)被使用,請換一個試試".encode('utf-8')) 
  13.         else
  14.             if not _from['name']: 
  15.                 sock.send(f"{name} 很高興見到你,現(xiàn)在可以暢游元宇宙了".encode('utf-8')) 
  16.                 msg = f"新成員{name} 加入了元宇宙,和TA聊聊吧".encode('utf-8'
  17.                 sendMsg(msg, _from) 
  18.             else
  19.                 sock.send(f"更換名稱完成".encode('utf-8')) 
  20.                 msg = f"{_from['name']} 更換名稱為 {name},和TA聊聊吧".encode('utf-8'
  21.                 sendMsg(msg, _from) 
  22.             _from['name'] = name 
  23.          
  24.     elif '@' in data: 
  25.         # 私信 
  26.         targets = re.findall(r'@(.+?) ', data) 
  27.         print(targets) 
  28.         msg = f"{_from['name']}: {data}".encode('utf-8'
  29.         sendMsg(msg, _from, targets) 
  30.     else
  31.         # 群信 
  32.         msg = f"{_from['name']}:{data}".encode('utf-8'
  33.         sendMsg(msg, _from) 
  • 代碼分為兩大部分,if 前面是處理收到的消息,將 bytes 轉(zhuǎn)化為 字符串;if 開始處理具體的消息
  • 如果收到 name: 開頭的消息,表示需要設(shè)置用戶名,其中包括判重,以及給其他成員發(fā)送消息
  • 如果收到的消息里有 @,表示在發(fā)私信,先提取出需要發(fā)出的用戶們,然后將消息發(fā)送給對應(yīng)的用戶
  • 如果沒有特殊標(biāo)記,就表示群發(fā)
  • 其中 sendMsg 用于發(fā)送消息,接收三個參數(shù),第一個是消息,第二是發(fā)送者,第三個是接收者名稱數(shù)組

當(dāng)鏈接關(guān)閉時,需要處理一下關(guān)閉的回調(diào)函數(shù):

  1. def onCloseConn(server, sock, ip): 
  2.     cid = f'{ip[0]}_{ip[1]}' 
  3.     name = clients[cid]['name'
  4.     if name
  5.         msg = f"{name} 從元宇宙中消失了".encode('utf-8'
  6.         sendMsg(msg, clients[cid]) 
  7.     del clients[cid] 
  • 當(dāng)收到鏈接斷開的消息時,合成消息,發(fā)送給其他用戶
  • 然后從客戶端緩存中刪除

客戶端

客戶端需要解決兩個問題,第一個是處理接收到的消息,第二個是允許用戶的輸入。

我們將接收消息作為一個線程,將用戶輸入作為主循環(huán)。

接收消息

先看接收消息的代碼:

  1. def receive(client): 
  2.     while True
  3.         try: 
  4.             s_info = client.recv(1024)  # 接受服務(wù)端的消息并解碼 
  5.             if not s_info: 
  6.                 print(f"{bcolors.WARNING}服務(wù)器鏈接斷開{bcolors.ENDC}"
  7.                 break 
  8.             print(f"{bcolors.OKCYAN}新的消息:{bcolors.ENDC}\n", bcolors.OKGREEN + s_info.decode('utf-8')+ bcolors.ENDC) 
  9.         except Exception: 
  10.             print(f"{bcolors.WARNING}服務(wù)器鏈接斷開{bcolors.ENDC}"
  11.             break 
  12.         if close
  13.             break 
  • 這是線程中用的代碼,接收一個客戶端鏈接作為參數(shù)
  • 在循環(huán)中不斷地從鏈接中獲取信息,如果沒有消息時 recv 方法會阻塞,直到有新的消息過來
  • 收到消息后,將消息寫出到控制臺上
  • bcolors 提供了一些顏色標(biāo)記,將消息顯示為不同的顏色
  • close 是一個全局標(biāo)記,如果客戶端需要退出時,會設(shè)置為 True,可以讓線程結(jié)束

輸入處理

下面再看一下輸入控制程序:

  1. while True
  2.     pass 
  3.     value = input(""
  4.     value = value.strip() 
  5.      
  6.     if value == ':start'
  7.         if thread: 
  8.             print(f"{bcolors.OKBLUE}您已經(jīng)在元宇宙中了{(lán)bcolors.ENDC}"
  9.         else
  10.             client = createClient(ip, 5000) 
  11.             thread = Thread(target=receive, args=(client,)) 
  12.             thread.start() 
  13.             print(f"{bcolors.OKBLUE}您進(jìn)入元宇宙了{(lán)bcolors.ENDC}"
  14.     elif value == ':quit' or value == ':stop'
  15.         if thread: 
  16.             client.close() 
  17.             close = True 
  18.             print(f"{bcolors.OKBLUE}正在退出中…{bcolors.ENDC}"
  19.             thread.join() 
  20.             print(f"{bcolors.OKBLUE}元宇宙已退出{bcolors.ENDC}"
  21.             thread = None 
  22.         if value == ':quit'
  23.             print(f"{bcolors.OKBLUE}退出程序{bcolors.ENDC}"
  24.             break 
  25.         pass 
  26.     elif value == ':help'
  27.         help() 
  28.     else
  29.         if client: 
  30.             # 聊天模式 
  31.             client.send(value.encode('utf-8')) 
  32.         else
  33.             print(f'{bcolors.WARNING}還沒接入元宇宙,請先輸入 :start 接入{bcolors.ENDC}'
  34.     client.close() 
  1. 主要是對不同的命令做出的相應(yīng),比如 :start 表示需要建立鏈接,:quit 表示退出等
  2. 命令前加 : 是為了和一般的消息做區(qū)分,如果不帶 : 就認(rèn)為是在發(fā)送消息

啟動

完成了整體編碼之后,就可以啟動了,最終的代碼由三部分組成。

第一部分是服務(wù)器端核心代碼,存放在 simpletcp.py 中。

第二部分是服務(wù)器端業(yè)務(wù)代碼,存放在 metaServer.py 中。

第三部分是客戶端代碼,存放在 metaClient.py 中。

另外需要一些輔助的處理,比如發(fā)送消息的 sendMsg 方法,顏色處理方法等,具體可以下載本文源碼了解。

進(jìn)入代碼目錄,啟動命令行,執(zhí)行 python metaServer.py,輸入指令 start:

server

然后再打開一個命令行,執(zhí)行 python metaClient.py,輸入指令 :start,就可以接入到元宇宙:

client

設(shè)置自己的名字:

如果有新的成員加入時,就會得到消息提醒, 還可以玩點互動:

怎么樣好玩吧,一個元宇宙就這樣形成了,趕緊讓其他伙伴加入試試吧。

總結(jié)

元宇宙現(xiàn)在是個很熱的概念,但還是基于現(xiàn)有的技術(shù)打造的,元宇宙給人們提供了一個生活在虛擬的神奇世界里的想象空間,其實自從有了互聯(lián)網(wǎng),我們就已經(jīng)逐步生活在元宇宙之中了。

今天我們用基礎(chǔ)的 TCP 技術(shù),構(gòu)建了一個自己的元宇宙聊天室,雖然功能上和想象中的元宇宙相去甚遠(yuǎn),不過其中的主要功能已經(jīng)成形了。

如果有興趣還可以在這個基礎(chǔ)上加入更好玩的功能,比如好友,群組,消息記錄等等,在深入了解的同時,讓這個元宇宙更好玩。

期望今天的你們元宇宙對你有所啟發(fā),歡迎在留言區(qū)寫下你的想法與觀點,比心!

參考代碼

https://github.com/JustDoPython/python-examples/tree/master/taiyangxue/meta

參考資料

[1]simpletcp: https://github.com/fschr/simpletcp

[2]select: https://docs.python.org/zh-cn/3/library/select.html

 

責(zé)任編輯:武曉燕 來源: Python技術(shù)
相關(guān)推薦

2021-10-18 09:09:16

數(shù)據(jù)庫

2023-02-03 17:50:29

元宇宙資本

2022-07-06 23:28:53

元宇宙Web3.0

2022-03-13 19:37:13

元宇宙安全隱私

2022-06-27 14:31:20

元宇宙品牌運(yùn)營推廣

2020-09-10 06:58:34

C語言DBProxy

2022-06-01 16:13:51

元宇宙

2023-06-28 14:37:20

元宇宙工業(yè)元宇宙

2020-04-30 14:25:13

代碼項目JS

2022-08-10 23:18:14

元宇宙NFT虛擬世界

2021-07-29 09:29:12

AI游戲DeepMind

2022-06-15 14:37:30

元宇宙

2022-04-28 13:56:10

元宇宙虛擬交易NFT

2022-06-29 13:46:50

元宇宙Web 3.0

2022-06-09 17:41:05

Lamina1元宇宙區(qū)塊鏈

2021-04-15 11:10:40

GitHub代碼開發(fā)者

2021-12-16 07:49:12

元宇宙VR平行時空

2021-11-18 11:01:03

元宇宙技術(shù)自動化

2022-07-14 14:05:26

元宇宙區(qū)塊鏈技術(shù)

2020-02-17 13:45:27

抓取代碼工具
點贊
收藏

51CTO技術(shù)棧公眾號