網(wǎng)絡(luò)編程

Socket是網(wǎng)絡(luò)編程的一個(gè)抽象概念。通常我們用一個(gè)Socket表示“打開(kāi)了一個(gè)網(wǎng)絡(luò)鏈接”,而打開(kāi)一個(gè)Socket需要知道目標(biāo)計(jì)算機(jī)的IP地址和端口號(hào),再指定協(xié)議類型即可。

TCP編程

客戶端

# 要?jiǎng)?chuàng)建一個(gè)基于TCP連接的Socket,可以這樣做:

# 導(dǎo)入socket庫(kù):
import socket
# 創(chuàng)建一個(gè)socket:
s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
# 建立連接:
s.connect(('www.sina.com.cn', 80))

# 創(chuàng)建Socket時(shí),AF_INET指定使用IPv4協(xié)議,如果要用更先進(jìn)的IPv6,就指定為AF_INET6。SOCK_STREAM指定使用面向流的TCP協(xié)議,這樣,一個(gè)Socket對(duì)象就創(chuàng)建成功,但是還沒(méi)有建立連接。
# 客戶端要主動(dòng)發(fā)起TCP連接,必須知道服務(wù)器的IP地址和端口號(hào)。新浪網(wǎng)站的IP地址可以用域名www.sina.com.cn自動(dòng)轉(zhuǎn)換到IP地址,但是怎么知道新浪服務(wù)器的端口號(hào)呢?
# 答案是作為服務(wù)器,提供什么樣的服務(wù),端口號(hào)就必須固定下來(lái)。由于我們想要訪問(wèn)網(wǎng)頁(yè),因此新浪提供網(wǎng)頁(yè)服務(wù)的服務(wù)器必須把端口號(hào)固定在80端口,因?yàn)?0端口是Web服務(wù)的標(biāo)準(zhǔn)端口。其他服務(wù)都有對(duì)應(yīng)的標(biāo)準(zhǔn)端口號(hào),例如SMTP服務(wù)是25端口,F(xiàn)TP服務(wù)是21端口,等等。端口號(hào)小于1024的是Internet標(biāo)準(zhǔn)服務(wù)的端口,端口號(hào)大于1024的,可以任意使用。
# 注意參數(shù)是一個(gè)tuple,包含地址和端口號(hào)。

# 建立TCP連接后,我們就可以向新浪服務(wù)器發(fā)送請(qǐng)求,要求返回首頁(yè)的內(nèi)容:
# 發(fā)送數(shù)據(jù):
s.send(b'GET / HTTP/1.1\r\nHost: www.sina.com.cn\r\nConnection: close\r\n\r\n')
# TCP連接創(chuàng)建的是雙向通道,雙方都可以同時(shí)給對(duì)方發(fā)數(shù)據(jù)。但是誰(shuí)先發(fā)誰(shuí)后發(fā),怎么協(xié)調(diào),要根據(jù)具體的協(xié)議來(lái)決定。例如,HTTP協(xié)議規(guī)定客戶端必須先發(fā)請(qǐng)求給服務(wù)器,服務(wù)器收到后才發(fā)數(shù)據(jù)給客戶端。

# 發(fā)送的文本格式必須符合HTTP標(biāo)準(zhǔn),如果格式?jīng)]問(wèn)題,接下來(lái)就可以接收新浪服務(wù)器返回的數(shù)據(jù)了:
# 接收數(shù)據(jù):
buffer = []
while True:
    # 每次最多接收1k字節(jié):
    d = s.recv(1024)
    if d:
        buffer.append(d)
    else:
        break
data = b''.join(buffer)
# 接收數(shù)據(jù)時(shí),調(diào)用recv(max)方法,一次最多接收指定的字節(jié)數(shù),因此,在一個(gè)while循環(huán)中反復(fù)接收,直到recv()返回空數(shù)據(jù),表示接收完畢,退出循環(huán)。

# 當(dāng)我們接收完數(shù)據(jù)后,調(diào)用close()方法關(guān)閉Socket,這樣,一次完整的網(wǎng)絡(luò)通信就結(jié)束了:
# 關(guān)閉連接:
s.close()

# 接收到的數(shù)據(jù)包括HTTP頭和網(wǎng)頁(yè)本身,我們只需要把HTTP頭和網(wǎng)頁(yè)分離一下,把HTTP頭打印出來(lái),網(wǎng)頁(yè)內(nèi)容保存到文件:

header, html = data.split(b'\r\n\r\n', 1)
print(header.decode('utf-8'))
# 把接收的數(shù)據(jù)寫(xiě)入文件:
with open('sina.html', 'wb') as f:
    f.write(html)
# 現(xiàn)在,只需要在瀏覽器中打開(kāi)這個(gè)sina.html文件,就可以看到新浪的首頁(yè)了。

服務(wù)器

# 服務(wù)器進(jìn)程首先要綁定一個(gè)端口并監(jiān)聽(tīng)來(lái)自其他客戶端的連接。如果某個(gè)客戶端連接過(guò)來(lái)了,服務(wù)器就與該客戶端建立Socket連接,隨后的通信就靠這個(gè)Socket連接了。
# 所以,服務(wù)器會(huì)打開(kāi)固定端口(比如80)監(jiān)聽(tīng),每來(lái)一個(gè)客戶端連接,就創(chuàng)建該Socket連接。由于服務(wù)器會(huì)有大量來(lái)自客戶端的連接,所以,服務(wù)器要能夠區(qū)分一個(gè)Socket連接是和哪個(gè)客戶端綁定的。一個(gè)Socket依賴4項(xiàng):服務(wù)器地址、服務(wù)器端口、客戶端地址、客戶端端口來(lái)唯一確定一個(gè)Socket。
# 但是服務(wù)器還需要同時(shí)響應(yīng)多個(gè)客戶端的請(qǐng)求,所以,每個(gè)連接都需要一個(gè)新的進(jìn)程或者新的線程來(lái)處理,否則,服務(wù)器一次就只能服務(wù)一個(gè)客戶端了。
# 我們來(lái)編寫(xiě)一個(gè)簡(jiǎn)單的服務(wù)器程序,它接收客戶端連接,把客戶端發(fā)過(guò)來(lái)的字符串加上Hello再發(fā)回去。
# 首先,創(chuàng)建一個(gè)基于IPv4和TCP協(xié)議的Socket:
s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)

# 然后,我們要綁定監(jiān)聽(tīng)的地址和端口。服務(wù)器可能有多塊網(wǎng)卡,可以綁定到某一塊網(wǎng)卡的IP地址上,也可以用0.0.0.0綁定到所有的網(wǎng)絡(luò)地址,還可以用127.0.0.1綁定到本機(jī)地址。127.0.0.1是一個(gè)特殊的IP地址,表示本機(jī)地址,如果綁定到這個(gè)地址,客戶端必須同時(shí)在本機(jī)運(yùn)行才能連接,也就是說(shuō),外部的計(jì)算機(jī)無(wú)法連接進(jìn)來(lái)。

# 端口號(hào)需要預(yù)先指定。因?yàn)槲覀儗?xiě)的這個(gè)服務(wù)不是標(biāo)準(zhǔn)服務(wù),所以用9999這個(gè)端口號(hào)。請(qǐng)注意,小于1024的端口號(hào)必須要有管理員權(quán)限才能綁定:
# 監(jiān)聽(tīng)端口:
s.bind(('127.0.0.1', 9999))

# 緊接著,調(diào)用listen()方法開(kāi)始監(jiān)聽(tīng)端口,傳入的參數(shù)指定等待連接的最大數(shù)量:
s.listen(5)
print('Waiting for connection...')
# 接下來(lái),服務(wù)器程序通過(guò)一個(gè)永久循環(huán)來(lái)接受來(lái)自客戶端的連接,accept()會(huì)等待并返回一個(gè)客戶端的連接:
while True:
    # 接受一個(gè)新連接:
    sock, addr = s.accept()
    # 創(chuàng)建新線程來(lái)處理TCP連接:
    t = threading.Thread(target=tcplink, args=(sock, addr))
    t.start()

# 每個(gè)連接都必須創(chuàng)建新線程(或進(jìn)程)來(lái)處理,否則,單線程在處理連接的過(guò)程中,無(wú)法接受其他客戶端的連接:
def tcplink(sock, addr):
    print('Accept new connection from %s:%s...' % addr)
    sock.send(b'Welcome!')
    while True:
        data = sock.recv(1024)
        time.sleep(1)
        if not data or data.decode('utf-8') == 'exit':
            break
        sock.send(('Hello, %s!' % data.decode('utf-8')).encode('utf-8'))
    sock.close()
    print('Connection from %s:%s closed.' % addr)

# 連接建立后,服務(wù)器首先發(fā)一條歡迎消息,然后等待客戶端數(shù)據(jù),并加上Hello再發(fā)送給客戶端。如果客戶端發(fā)送了exit字符串,就直接關(guān)閉連接。

# 要測(cè)試這個(gè)服務(wù)器程序,我們還需要編寫(xiě)一個(gè)客戶端程序:
s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
# 建立連接:
s.connect(('127.0.0.1', 9999))
# 接收歡迎消息:
print(s.recv(1024).decode('utf-8'))
for data in [b'Michael', b'Tracy', b'Sarah']:
    # 發(fā)送數(shù)據(jù):
    s.send(data)
    print(s.recv(1024).decode('utf-8'))
s.send(b'exit')
s.close()

UDP編程

使用UDP協(xié)議時(shí),不需要建立連接,只需要知道對(duì)方的IP地址和端口號(hào),就可以直接發(fā)數(shù)據(jù)包。

# 服務(wù)器首先需要綁定端口:
s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
# 綁定端口:
s.bind(('127.0.0.1', 9999))

# 創(chuàng)建Socket時(shí),SOCK_DGRAM指定了這個(gè)Socket的類型是UDP。綁定端口和TCP一樣,但是不需要調(diào)用listen()方法,而是直接接收來(lái)自任何客戶端的數(shù)據(jù):
print('Bind UDP on 9999...')
while True:
    # 接收數(shù)據(jù):
    data, addr = s.recvfrom(1024)
    print('Received from %s:%s.' % addr)
    s.sendto(b'Hello, %s!' % data, addr)
# recvfrom()方法返回?cái)?shù)據(jù)和客戶端的地址與端口,這樣,服務(wù)器收到數(shù)據(jù)后,直接調(diào)用sendto()就可以把數(shù)據(jù)用UDP發(fā)給客戶端。

# 客戶端使用UDP時(shí),首先仍然創(chuàng)建基于UDP的Socket,然后,不需要調(diào)用connect(),直接通過(guò)sendto()給服務(wù)器發(fā)數(shù)據(jù):
s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
for data in [b'Michael', b'Tracy', b'Sarah']:
    # 發(fā)送數(shù)據(jù):
    s.sendto(data, ('127.0.0.1', 9999))
    # 接收數(shù)據(jù):
    print(s.recv(1024).decode('utf-8'))
s.close()
# 從服務(wù)器接收數(shù)據(jù)仍然調(diào)用recv()方法。
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書(shū)系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 230,622評(píng)論 6 544
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 99,716評(píng)論 3 429
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人,你說(shuō)我怎么就攤上這事。” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 178,746評(píng)論 0 383
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我,道長(zhǎng),這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 63,991評(píng)論 1 318
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 72,706評(píng)論 6 413
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 56,036評(píng)論 1 329
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼。 笑死,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 44,029評(píng)論 3 450
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 43,203評(píng)論 0 290
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 49,725評(píng)論 1 336
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 41,451評(píng)論 3 361
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 43,677評(píng)論 1 374
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 39,161評(píng)論 5 365
  • 正文 年R本政府宣布,位于F島的核電站,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 44,857評(píng)論 3 351
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 35,266評(píng)論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 36,606評(píng)論 1 295
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 52,407評(píng)論 3 400
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 48,643評(píng)論 2 380

推薦閱讀更多精彩內(nèi)容

  • 原文轉(zhuǎn)自:Python教程 TCP/IP簡(jiǎn)介雖然大家現(xiàn)在對(duì)互聯(lián)網(wǎng)很熟悉,但是計(jì)算機(jī)網(wǎng)絡(luò)的出現(xiàn)比互聯(lián)網(wǎng)要早很多。 計(jì)...
    李牧羊閱讀 936評(píng)論 0 5
  • 網(wǎng)絡(luò)編程 網(wǎng)絡(luò)編程對(duì)所有開(kāi)發(fā)語(yǔ)言都是一樣的,Python也不例外。用Python進(jìn)行網(wǎng)絡(luò)編程,就是在Python程...
    時(shí)間之友閱讀 128評(píng)論 0 0
  • 1. 網(wǎng)絡(luò)編程概述 1.1 計(jì)算機(jī)網(wǎng)絡(luò) 是指將地理位置不同的具有獨(dú)立功能的多臺(tái)計(jì)算機(jī)及其外部設(shè)備,通過(guò)通信線路連接...
    JackChen1024閱讀 1,048評(píng)論 0 3
  • 1 UDP 1什么是網(wǎng)絡(luò) ·網(wǎng)絡(luò)就是一種輔助雙方或者多方能夠連接在一起的工具 ·如果沒(méi)有網(wǎng)絡(luò)可想單機(jī)的世界是多么的...
    五行缺覺(jué)閱讀 418評(píng)論 0 1
  • 每次和你相遇 輕觸歲月拋下的墨跡 贊嘆時(shí)光這個(gè)偉大的機(jī)器 寸寸纏繞,絲絲嵌入 讓兩個(gè)擦肩而過(guò)的靈魂 交融 相匯 ...
    雅爾達(dá)的朝陽(yáng)閱讀 435評(píng)論 0 0