快速進(jìn)修Python指南之網(wǎng)絡(luò)編程及并發(fā)編程
引言
今天我們將對(duì)網(wǎng)絡(luò)編程和多線程技術(shù)進(jìn)行講解,這兩者的原理大家都已經(jīng)了解了,因此我們主要關(guān)注的是它們的寫(xiě)法區(qū)別。雖然這些區(qū)別并不是非常明顯,但我們之所以將網(wǎng)絡(luò)編程和多線程一起講解,是因?yàn)樵趯W(xué)習(xí)Java的socket知識(shí)時(shí),我們通常會(huì)將它們結(jié)合使用,以實(shí)現(xiàn)服務(wù)器對(duì)多個(gè)客戶(hù)端連接的阻塞IO的處理。雖然我是這樣解釋的,但是Python在控制連接數(shù)方面更加友好,相對(duì)于Java來(lái)說(shuō)更加便捷。好了,廢話不多說(shuō),讓我們開(kāi)始今天的講解吧。
socket及線程
這里我將給大家舉一個(gè)例子,同時(shí)也會(huì)指出一些需要注意的問(wèn)題,以幫助Java同學(xué)們避免再次遇到這些坑。
import socket
import multiprocessing
import time
# 創(chuàng)建互斥鎖
lock = multiprocessing.Lock()
# 處理客戶(hù)端請(qǐng)求的函數(shù)
def handle_client(conn, addr):
print(f"Connected to {addr}")
time.sleep(100)
# 創(chuàng)建服務(wù)器
def create_server():
# 創(chuàng)建socket對(duì)象
server_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
# 綁定地址和端口
server_addr = ("localhost", 8000)
server_socket.bind(server_addr)
# 監(jiān)聽(tīng)連接
server_socket.listen(1)
print("Server started. Listening for connections...")
while True:
# 接受客戶(hù)端連接
conn, addr = server_socket.accept()
handle_client(conn, addr)
# 創(chuàng)建進(jìn)程處理客戶(hù)端請(qǐng)求
process = multiprocessing.Process(target=handle_client, args=(conn, addr))
process.start()
if __name__ == '__main__':
# 啟動(dòng)服務(wù)器
print("啟動(dòng)服務(wù)器")
create_server()以下是客戶(hù)端的代碼
import socket
import time
client = socket.socket() #創(chuàng)建socket對(duì)象
host = '127.0.0.1' #服務(wù)端ip
port = 8000 #服務(wù)端ip端口
client.connect((host, port)) #根據(jù)服務(wù)端地址,建立連接
print('client對(duì)象:', client) #查看socket對(duì)象屬性
time.sleep(100)
#client.close() #關(guān)閉與服務(wù)端的連接上面的例子已經(jīng)涵蓋了我今天要講的內(nèi)容,所以沒(méi)有太多需要補(bǔ)充的了。
與Java的一些區(qū)別
首先,Python使用multiprocessing來(lái)創(chuàng)建多線程,當(dāng)然還有其他的包可以實(shí)現(xiàn)相同的功能,這里就不一一贅述了。另外,還有一個(gè)需要注意的地方是,在Python中使用if __name__ == '__main__':語(yǔ)句時(shí),你必須將其寫(xiě)在主函數(shù)中,而不要寫(xiě)在你定義的子函數(shù)中。為什么要這樣寫(xiě)呢?原因是,當(dāng)調(diào)用process.start()后,Python會(huì)重新執(zhí)行當(dāng)前文件,也就是說(shuō)如果你將if __name__ == '__main__':這句話寫(xiě)在非主函數(shù)中,create_server()它將會(huì)被再次調(diào)用。而使用if __name__ == '__main__':語(yǔ)句可以判斷是否是主函數(shù)執(zhí)行,如果不是,則會(huì)過(guò)濾掉這部分代碼。雖然這個(gè)機(jī)制可能有些令人困惑,但是去深入了解其執(zhí)行原理并不是必要的,所以大家要記住這個(gè)要點(diǎn)。這張圖顯示的是再次被調(diào)用是的名字:
第二:在Python官網(wǎng)中指出,通過(guò)調(diào)用server_socket.listen(1)可以啟動(dòng)一個(gè)服務(wù)器,用于接受連接并將未接受的客戶(hù)端連接放入等待隊(duì)列中。需要注意的是,等待隊(duì)列的大小由listen(n)中的參數(shù)n+1指定,并不代表實(shí)際監(jiān)聽(tīng)到的客戶(hù)端連接。如果超過(guò)隊(duì)列大小的連接嘗試進(jìn)入,服務(wù)器將直接報(bào)錯(cuò)。
總結(jié)
今天我們學(xué)習(xí)了網(wǎng)絡(luò)編程和多線程技術(shù)的寫(xiě)法區(qū)別。我們主要關(guān)注了在Java中使用socket和多線程結(jié)合實(shí)現(xiàn)服務(wù)器處理多個(gè)客戶(hù)端連接的阻塞IO的方法,以及在Python中使用multiprocessing模塊創(chuàng)建多線程的方式。通過(guò)一個(gè)實(shí)例來(lái)說(shuō)明了這些概念,并指出了需要注意的問(wèn)題。其實(shí)了解了這些基本用法后,我們還能夠自己實(shí)現(xiàn)許多其他功能,例如了解了線程之后,就知道會(huì)有隊(duì)列的概念,然后可以嘗試自己實(shí)現(xiàn)一個(gè)生產(chǎn)者消費(fèi)者隊(duì)列。這與學(xué)習(xí)Java的路線非常相似,并且我們擁有豐富的開(kāi)發(fā)經(jīng)驗(yàn),因此我們只需要關(guān)注語(yǔ)法方面的學(xué)習(xí)即可~~
以上就是Java開(kāi)發(fā)者如何快速進(jìn)修Python指南之網(wǎng)絡(luò)編程及并發(fā)編程的詳細(xì)內(nèi)容,更多關(guān)于Python并發(fā)網(wǎng)絡(luò)編程的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Python利用shutil實(shí)現(xiàn)拷貝文件功能
shutil?是一個(gè)?Python?內(nèi)置模塊,該模塊對(duì)文件的復(fù)制、刪除和壓縮等操作都提供了非常方便的支持。本文將利用shutil實(shí)現(xiàn)拷貝文件功能,需要的可以參考一下2022-07-07
python密碼學(xué)換位密碼及換位解密轉(zhuǎn)置加密教程
這篇文章主要為大家介紹了python密碼學(xué)換位密碼及換位解密轉(zhuǎn)置加密教程,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-05-05
python遠(yuǎn)程連接MySQL數(shù)據(jù)庫(kù)
這篇文章主要為大家詳細(xì)介紹了python遠(yuǎn)程連接MySQL數(shù)據(jù)庫(kù),拉取數(shù)據(jù)存至本地文件,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-04-04
python shutil文件操作工具使用實(shí)例分析
這篇文章主要介紹了python shutil文件操作工具使用實(shí)例分析,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-12-12
python 數(shù)據(jù)分析實(shí)現(xiàn)長(zhǎng)寬格式的轉(zhuǎn)換
這篇文章主要介紹了python 數(shù)據(jù)分析實(shí)現(xiàn)長(zhǎng)寬格式的轉(zhuǎn)換,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-05-05
python字典與json轉(zhuǎn)換的方法總結(jié)
在本篇文章里小編給大家整理的是一篇關(guān)于python字典與json轉(zhuǎn)換的方法總結(jié)內(nèi)容,有需要的朋友們可以學(xué)習(xí)下。2020-12-12

