網(wǎng)絡編程

Socket是網(wǎng)絡編程的一個抽象概念庆锦。通常我們用一個Socket表示“打開了一個網(wǎng)絡鏈接”捅位,而打開一個Socket需要知道目標計算機的IP地址和端口號,再指定協(xié)議類型即可搂抒。

TCP編程

客戶端

# 要創(chuàng)建一個基于TCP連接的Socket艇搀,可以這樣做:

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

# 創(chuàng)建Socket時,AF_INET指定使用IPv4協(xié)議求晶,如果要用更先進的IPv6焰雕,就指定為AF_INET6。SOCK_STREAM指定使用面向流的TCP協(xié)議誉帅,這樣淀散,一個Socket對象就創(chuàng)建成功,但是還沒有建立連接蚜锨。
# 客戶端要主動發(fā)起TCP連接档插,必須知道服務器的IP地址和端口號。新浪網(wǎng)站的IP地址可以用域名www.sina.com.cn自動轉換到IP地址亚再,但是怎么知道新浪服務器的端口號呢郭膛?
# 答案是作為服務器,提供什么樣的服務氛悬,端口號就必須固定下來则剃。由于我們想要訪問網(wǎng)頁,因此新浪提供網(wǎng)頁服務的服務器必須把端口號固定在80端口如捅,因為80端口是Web服務的標準端口棍现。其他服務都有對應的標準端口號,例如SMTP服務是25端口镜遣,F(xiàn)TP服務是21端口己肮,等等。端口號小于1024的是Internet標準服務的端口悲关,端口號大于1024的谎僻,可以任意使用。
# 注意參數(shù)是一個tuple寓辱,包含地址和端口號艘绍。

# 建立TCP連接后,我們就可以向新浪服務器發(fā)送請求秫筏,要求返回首頁的內(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)建的是雙向通道诱鞠,雙方都可以同時給對方發(fā)數(shù)據(jù)挎挖。但是誰先發(fā)誰后發(fā),怎么協(xié)調(diào)般甲,要根據(jù)具體的協(xié)議來決定肋乍。例如鹅颊,HTTP協(xié)議規(guī)定客戶端必須先發(fā)請求給服務器敷存,服務器收到后才發(fā)數(shù)據(jù)給客戶端。

# 發(fā)送的文本格式必須符合HTTP標準堪伍,如果格式?jīng)]問題锚烦,接下來就可以接收新浪服務器返回的數(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ù)時,調(diào)用recv(max)方法帝雇,一次最多接收指定的字節(jié)數(shù)涮俄,因此,在一個while循環(huán)中反復接收尸闸,直到recv()返回空數(shù)據(jù)彻亲,表示接收完畢,退出循環(huán)吮廉。

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

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

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

服務器

# 服務器進程首先要綁定一個端口并監(jiān)聽來自其他客戶端的連接注益。如果某個客戶端連接過來了,服務器就與該客戶端建立Socket連接溯捆,隨后的通信就靠這個Socket連接了丑搔。
# 所以,服務器會打開固定端口(比如80)監(jiān)聽现使,每來一個客戶端連接低匙,就創(chuàng)建該Socket連接。由于服務器會有大量來自客戶端的連接碳锈,所以顽冶,服務器要能夠區(qū)分一個Socket連接是和哪個客戶端綁定的。一個Socket依賴4項:服務器地址售碳、服務器端口强重、客戶端地址绞呈、客戶端端口來唯一確定一個Socket。
# 但是服務器還需要同時響應多個客戶端的請求间景,所以佃声,每個連接都需要一個新的進程或者新的線程來處理,否則倘要,服務器一次就只能服務一個客戶端了圾亏。
# 我們來編寫一個簡單的服務器程序,它接收客戶端連接封拧,把客戶端發(fā)過來的字符串加上Hello再發(fā)回去志鹃。
# 首先,創(chuàng)建一個基于IPv4和TCP協(xié)議的Socket:
s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)

# 然后泽西,我們要綁定監(jiān)聽的地址和端口曹铃。服務器可能有多塊網(wǎng)卡,可以綁定到某一塊網(wǎng)卡的IP地址上捧杉,也可以用0.0.0.0綁定到所有的網(wǎng)絡地址陕见,還可以用127.0.0.1綁定到本機地址。127.0.0.1是一個特殊的IP地址味抖,表示本機地址评甜,如果綁定到這個地址,客戶端必須同時在本機運行才能連接非竿,也就是說蜕着,外部的計算機無法連接進來。

# 端口號需要預先指定红柱。因為我們寫的這個服務不是標準服務承匣,所以用9999這個端口號。請注意锤悄,小于1024的端口號必須要有管理員權限才能綁定:
# 監(jiān)聽端口:
s.bind(('127.0.0.1', 9999))

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

# 每個連接都必須創(chuàng)建新線程(或進程)來處理,否則隶症,單線程在處理連接的過程中政模,無法接受其他客戶端的連接:
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)

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

# 要測試這個服務器程序刊咳,我們還需要編寫一個客戶端程序:
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é)議時,不需要建立連接儡司,只需要知道對方的IP地址和端口號娱挨,就可以直接發(fā)數(shù)據(jù)包。

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

# 創(chuàng)建Socket時捕犬,SOCK_DGRAM指定了這個Socket的類型是UDP跷坝。綁定端口和TCP一樣,但是不需要調(diào)用listen()方法或听,而是直接接收來自任何客戶端的數(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()方法返回數(shù)據(jù)和客戶端的地址與端口探孝,這樣,服務器收到數(shù)據(jù)后誉裆,直接調(diào)用sendto()就可以把數(shù)據(jù)用UDP發(fā)給客戶端。

# 客戶端使用UDP時缸濒,首先仍然創(chuàng)建基于UDP的Socket足丢,然后,不需要調(diào)用connect()庇配,直接通過sendto()給服務器發(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()
# 從服務器接收數(shù)據(jù)仍然調(diào)用recv()方法斩跌。
最后編輯于
?著作權歸作者所有,轉載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市捞慌,隨后出現(xiàn)的幾起案子耀鸦,更是在濱河造成了極大的恐慌,老刑警劉巖啸澡,帶你破解...
    沈念sama閱讀 222,627評論 6 517
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件袖订,死亡現(xiàn)場離奇詭異,居然都是意外死亡嗅虏,警方通過查閱死者的電腦和手機洛姑,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 95,180評論 3 399
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來皮服,“玉大人楞艾,你說我怎么就攤上這事×涔悖” “怎么了硫眯?”我有些...
    開封第一講書人閱讀 169,346評論 0 362
  • 文/不壞的土叔 我叫張陵,是天一觀的道長择同。 經(jīng)常有香客問我两入,道長,這世上最難降的妖魔是什么奠衔? 我笑而不...
    開封第一講書人閱讀 60,097評論 1 300
  • 正文 為了忘掉前任谆刨,我火速辦了婚禮塘娶,結果婚禮上,老公的妹妹穿的比我還像新娘痊夭。我一直安慰自己刁岸,他們只是感情好,可當我...
    茶點故事閱讀 69,100評論 6 398
  • 文/花漫 我一把揭開白布她我。 她就那樣靜靜地躺著虹曙,像睡著了一般。 火紅的嫁衣襯著肌膚如雪番舆。 梳的紋絲不亂的頭發(fā)上酝碳,一...
    開封第一講書人閱讀 52,696評論 1 312
  • 那天,我揣著相機與錄音恨狈,去河邊找鬼疏哗。 笑死,一個胖子當著我的面吹牛禾怠,可吹牛的內(nèi)容都是我干的返奉。 我是一名探鬼主播,決...
    沈念sama閱讀 41,165評論 3 422
  • 文/蒼蘭香墨 我猛地睜開眼吗氏,長吁一口氣:“原來是場噩夢啊……” “哼芽偏!你這毒婦竟也來了?” 一聲冷哼從身側響起弦讽,我...
    開封第一講書人閱讀 40,108評論 0 277
  • 序言:老撾萬榮一對情侶失蹤污尉,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后往产,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體被碗,經(jīng)...
    沈念sama閱讀 46,646評論 1 319
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 38,709評論 3 342
  • 正文 我和宋清朗相戀三年捂齐,在試婚紗的時候發(fā)現(xiàn)自己被綠了蛮放。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 40,861評論 1 353
  • 序言:一個原本活蹦亂跳的男人離奇死亡奠宜,死狀恐怖包颁,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情压真,我是刑警寧澤娩嚼,帶...
    沈念sama閱讀 36,527評論 5 351
  • 正文 年R本政府宣布,位于F島的核電站滴肿,受9級特大地震影響岳悟,放射性物質發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 42,196評論 3 336
  • 文/蒙蒙 一贵少、第九天 我趴在偏房一處隱蔽的房頂上張望呵俏。 院中可真熱鬧,春花似錦滔灶、人聲如沸普碎。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,698評論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽麻车。三九已至,卻和暖如春斗这,著一層夾襖步出監(jiān)牢的瞬間动猬,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,804評論 1 274
  • 我被黑心中介騙來泰國打工表箭, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留赁咙,地道東北人。 一個月前我還...
    沈念sama閱讀 49,287評論 3 379
  • 正文 我出身青樓燃逻,卻偏偏與公主長得像序目,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子伯襟,可洞房花燭夜當晚...
    茶點故事閱讀 45,860評論 2 361

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

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