python利用socket實現(xiàn)客戶端和服務(wù)端之間進行通信
前言:
今天教大家通過Python進行Socket網(wǎng)絡(luò)編程(做一個聊天程序),可以實現(xiàn)在不同的主機(電腦)之間進行通話。
具體效果如何,接著往下看:
可以看到客戶端(上方)向服務(wù)器端(下方)發(fā)送了內(nèi)容,服務(wù)器端進行了回復(fù)
【備注:客戶端是我的本機,服務(wù)器是另一條主機(阿里云服務(wù)器)】
兩臺主機的目的:驗證兩臺主機可以相互通信
一、socket
先簡單給大家介紹一下什么是socket,socket(簡稱 套接字) 是進程間通信的一種方式,它與其他進程間通信的一個主要不同是:它能實現(xiàn)不同主機間的進程間通信。
我們網(wǎng)絡(luò)上各種各樣的服務(wù)大多都是基于 Socket 來完成通信的,例如瀏覽網(wǎng)頁、QQ 聊天、收發(fā) email 等等
簡單的說:socket可以實現(xiàn)不同主機間進行通信
socket通信的條件:IP和端口
ip相信大家都陌生了,每一臺主機都有一個ip,不同主機之間通信的首要前提就是ip可以互訪,此外還有一個條件就是端口,比如我們經(jīng)常聽到的80端口,3306端口,8080端口等。
主機中的數(shù)據(jù)是通過端口發(fā)送和接收,需要將對應(yīng)端口打開才能進行通信。
形象比喻
ip相當于家庭地址,端口相當于門或者窗戶
例子:
(主機A)快遞員要想將快遞(數(shù)據(jù))送到你手中(另一臺主機B),需要知道你家的地址(主機B的ip),到你家門口后,需要你打開門(主機B的端口)才能拿到快遞(數(shù)據(jù))。
這里需要分服務(wù)端和客戶端,客戶端發(fā)送(主機A),服務(wù)器接收(主機B),當然了,每一臺主機可以充當兩個角色(既是客戶端,也是服務(wù)器),這樣就可以實現(xiàn)兩臺主機之間相互發(fā)送和接收。
看到這里之后,相信大家都清楚socket在實現(xiàn)不同主機之間通信的大概意思了,下面開始Python代碼實現(xiàn)。
二、客戶端實現(xiàn)過程
先來分析客戶端(主機A)的實現(xiàn)過程:
from socket import * # 1.創(chuàng)建套接字 tcp_socket = socket(AF_INET,SOCK_STREAM) # 2.準備連接服務(wù)器,建立連接 serve_ip = "服務(wù)器端(主機B)的IP" serve_port = 8000 #端口,比如8000 tcp_socket.connect((serve_ip,serve_port)) # 連接服務(wù)器,建立連接,參數(shù)是元組形式
首先與服務(wù)器接收端(主機B)建立連接,連接條件(主機B的ip和端口),這里的端口8000是指將數(shù)據(jù)發(fā)送到主機B的端口(主機B到時候會監(jiān)聽8000端口,然后進行接收數(shù)據(jù))
#準備需要傳送的數(shù)據(jù) send_data = "今天是2021年08月29日,辰哥給服務(wù)器端發(fā)送數(shù)據(jù)了" tcp_socket.send(send_data.encode("gbk")) #從服務(wù)器接收數(shù)據(jù) #注意這個1024byte,大小根據(jù)需求自己設(shè)置 from_server_msg = tcp_socket.recv(1024) #加上.decode("gbk")可以解決亂碼 print(from_server_msg.decode("gbk")) #關(guān)閉連接 tcp_socket.close()
send_data是往服務(wù)器端(主機B)發(fā)送的內(nèi)容,from_server_msg是服務(wù)器端(主機B)往客戶端(主機A)發(fā)送的內(nèi)容
客戶端的代碼就結(jié)束了
三、服務(wù)器實現(xiàn)過程
分析服務(wù)器端(主機B)的實現(xiàn)過程:
from socket import * #創(chuàng)建套接字 tcp_server = socket(AF_INET,SOCK_STREAM) #綁定ip,port #這里ip默認本機 address = ('',8000) tcp_server.bind(address) # 啟動被動連接 #多少個客戶端可以連接 tcp_server.listen(128) #使用socket創(chuàng)建的套接字默認的屬性是主動的 #使用listen將其變?yōu)楸粍拥?,這樣就可以接收別人的鏈接了
服務(wù)器端(主機B)ip可以留空(默認本機),端口8000(因為客戶端往8000端口發(fā)送數(shù)據(jù),所以服務(wù)器需要監(jiān)聽的端口也是8000,與客戶端的端口一致)
# 創(chuàng)建接收 # 如果有新的客戶端來鏈接服務(wù)器,那么就產(chǎn)生一個新的套接字專門為這個客戶端服務(wù) client_socket, clientAddr = tcp_server.accept()
client_socket用來為這個客戶端服務(wù),相當于的tcp_server套接字的代理 tcp_server_socket就可以省下來專門等待其他新客戶端的鏈接 這里clientAddr存放的就是連接服務(wù)器的客戶端地址 #接收對方發(fā)送過來的數(shù)據(jù) from_client_msg = client_socket.recv(1024)#接收1024給字節(jié),這里recv接收的不再是元組,區(qū)別UDP print("接收的數(shù)據(jù):",from_client_msg.encode("gbk")) #發(fā)送數(shù)據(jù)給客戶端 send_data = client_socket.send("客戶端你好,服務(wù)器端收到,公眾號【Python研究者】".encode("gbk")) #關(guān)閉套接字 #關(guān)閉為這個客戶端服務(wù)的套接字,就意味著為不能再為這個客戶端服務(wù)了 #如果還需要服務(wù),只能再次重新連 client_socket.close()
from_client_msgs 是服務(wù)器端(主機B)接收到來自客戶端(主機A)發(fā)送過來的數(shù)據(jù)send_data 是服務(wù)器端(主機B)往客戶端(主機A)發(fā)送過去的數(shù)據(jù)
服務(wù)器端的代碼就結(jié)束了
提醒:服務(wù)器端的8000端口需要開啟,不然無法進行通信
四、演示
先啟動(執(zhí)行)服務(wù)器端(主機B)的程序,再執(zhí)行客戶端(主機A)
可以看到客戶端(上方)向服務(wù)器端(下方)發(fā)送了內(nèi)容,服務(wù)器端進行了回復(fù)
發(fā)送和響應(yīng)內(nèi)容:
客戶端發(fā)送:今天是2021年08月29日,辰哥給服務(wù)器端發(fā)送數(shù)據(jù)了
服務(wù)器端接收并回復(fù)給客戶端:客戶端你好,服務(wù)器端收到,公眾號【Python研究者】
五、實現(xiàn)持續(xù)通信過程
上方動圖演示的是客戶端和服務(wù)端的一次通信過程,可以將客戶端的發(fā)送和服務(wù)端的接收放到循環(huán)中,實現(xiàn)持續(xù)通信過程。
客戶端:
while(1): send_data = input("請輸入內(nèi)容:") #send_data = "今天是2021年08月29日,辰哥給服務(wù)器端發(fā)送數(shù)據(jù)了" tcp_socket.send(send_data.encode("gbk")) if send_data == "exit": break; #從服務(wù)器接收數(shù)據(jù) #注意這個1024byte,大小根據(jù)需求自己設(shè)置 from_server_msg = tcp_socket.recv(1024) #加上.decode("gbk")可以解決亂碼 print(from_server_msg.decode("gbk")) #關(guān)閉連接 tcp_socket.close()
服務(wù)端:
while(1): #接收對方發(fā)送過來的數(shù)據(jù) from_client_msg = client_socket.recv(1024)#接收1024給字節(jié),這里recv接收的不再是元組,區(qū)別UDP if(from_client_msg=="exit"): break print("接收的數(shù)據(jù):",from_client_msg.decode("gbk")) now_time = time.strftime('%Y-%m-%d %H:%M:%S', time.localtime(time.time())) #發(fā)送數(shù)據(jù)給客戶端 send_data = client_socket.send((str(now_time)+" 服務(wù)端:客戶端你好,服務(wù)器端收到,公眾號【Python研究者】").encode("gbk")) #關(guān)閉套接字 #關(guān)閉為這個客戶端服務(wù)的套接字,就意味著為不能再為這個客戶端服務(wù)了 #如果還需要服務(wù),只能再次重新連 client_socket.close()
客戶端可以持續(xù)給服務(wù)端發(fā)送數(shù)據(jù),服務(wù)器接收數(shù)據(jù)后打印并返回數(shù)據(jù)給客戶端
服務(wù)端返回的內(nèi)容:
當前系統(tǒng)時間+服務(wù)端:客戶端你好,服務(wù)器端收到
最后當客戶端輸入:exit,則斷開與服務(wù)端的連接:
到此這篇關(guān)于python利用socket實現(xiàn)客戶端和服務(wù)端之間進行通信 的文章就介紹到這了,更多相關(guān)python主機通信內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
python實現(xiàn)帶聲音的摩斯碼翻譯實現(xiàn)方法
這篇文章主要介紹了python實現(xiàn)帶聲音的摩斯碼翻譯實現(xiàn)方法,涉及pygame模塊操作及摩斯碼實現(xiàn)技巧,需要的朋友可以參考下2015-05-05淺談cv2.imread()和keras.preprocessing中的image.load_img()區(qū)別
這篇文章主要介紹了淺談cv2.imread()和keras.preprocessing中的image.load_img()區(qū)別說明,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-06-06Python functools.lru_cache裝飾器性能提升利器深入探究
本文將詳細介紹functools.lru_cache裝飾器的原理、用法以及適當?shù)膱鼍?以幫助你更好地利用這一功能,它可以用來緩存函數(shù)的輸出,以避免重復(fù)計算,從而顯著提高程序的執(zhí)行速度2024-01-01對python while循環(huán)和雙重循環(huán)的實例詳解
今天小編就為大家分享一篇對python while循環(huán)和雙重循環(huán)的實例詳解,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2019-08-08