1.在開始之前
1.1 TCP/IP協議
1.1.1 TCP/IP概述
在互聯網通信中,計算機之間的通信需要依賴互聯網通信協議,常見的網路通信協議有TCP/IP協議、IPX/SPX協議、NetBEUI協議等,TCP/IP就是其中一個網路通信協議,而我們今天要利用的socket就是依賴TCP/IP完成通信的,
實際上,TCP/IP并不是單獨一個協議,而是一個由FTP、SMTP、TCP、UDP、IP等協議構成的協議簇, 只是因為在TCP/IP協議中TCP協議和IP協議最具代表性,所以被稱為TCP/IP協議,
在網路通信協議中,TCP/IP的優勢在于具有很強的靈活性,支持任意規模的網路,幾乎可連接所有服務器和作業站,在使用TCP/IP協議時需要進行復雜的設定,每個結點至少需要一個“IP地址”、一個“子網掩碼”、一個“默認網關”、一個“主機名”,
1.1.2 TCP/IP協議組成
TCP/IP在一定程度上參考了OSI模型和DoD模型,所以其組成一定程度上與二者十分相似,而在TCP/IP中,分層被分為四個部分,分別是應用層、傳輸層、網路層和資料鏈路層,
下圖展示了四個部分分別對應的TCP/IP協議簇中的幾個不同協議
1.應用層
應用層是TCP/IP中的第一層,是直接為應用行程提供服務的,對不同種類的應用程式它們會根據自己的需要來使用應用層的不同協議,比如郵件使用的SMTP協議,萬維網的http協議,
應用層能加密、解密、格式化資料,還可以建立或解除與其他節點的聯系,這樣可以充分節省網路資源,
2.運輸層
運輸層是TCP/IP的核心層,TCP協議的UDP協議共同組成了TCP/IP的運輸層
3.網路層
網路層:網路層在TCP/IP協議中的位于第三層,在TCP/IP協議中網路層可以進行網路連接的建立和終止以及IP地址的尋找等功能
4.網路介面層
在TCP/IP協議中,網路介面層位于第四層,由于網路介面層兼并了物理層和資料鏈路層,所以,網路介面層既是傳輸資料的物理媒介,也可以為網路層提供一條準確無誤的線路,
1.1.3 TCP協議
TCP即傳輸控制協議,是一種面向連接的、可靠的、基于位元組流的通信協議,簡單來說TCP就是有確認機制的UDP協議,每發出一個資料包都要求確認,如果有一個資料包丟失,就收不到確認,發送方就必須重發這個資料包,為了保證傳輸的可靠性,
TCP協議在UDP基礎之上建立了三次對話的確認機制,即在正式收發資料前,必須和對方建立可靠的連接,TCP資料包和UDP一樣,都是由首部和資料兩部分組成,唯一不同的是,TCP資料包沒有長度限制,理論上可以無限長,但是為了保證網路的效率,通常TCP資料包的長度不會超過IP資料包的長度,以確保單個TCP資料包不必再分割,
1.2 socket套接字
1.2.1 socket概述
套接字是通信的基石,是支持TCP/IP協議的路通信的基本操作單元,
可以將套接字看作不同主機間的行程進行雙間通信的端點,它構成了單個主機內及整個網路間的編程界面,套接字存在于通信域中,通信域是為了處理一般的執行緒通過套接字通信而引進的一種抽象概念,套接字通常和同一個域中的套接字交換資料(資料交換也可能穿越域的界限,但這時一定要執行某種解釋程式),各種行程使用這個相同的域互相之間用Internet協議簇來進行通信
1.2.2 socket作業流程
1.服務器監聽
服務器首先啟動監聽埠動態,監聽埠的連接請求
2.客戶端發起請求
客戶端將向目標Ip的指定埠發起請求,為此,客戶端的套接字必須首先描述它要連接的服務器的套接字,指出服務器端套接字的地址和埠號,然后就向服務器端接字提出連接請求
3.連接確認,建立連接
當服務器端套接字監聽到或者說接收到客戶端套接字的連接請求,就會回應客戶端套接字的請求,建立一個新的執行緒,并把服務器端套接字的描述發送給客戶端,一旦客戶端確認了此描述,連接就建立好了,而服務器端套接字繼續處于監聽狀態,接收其他客戶端套接字的連接請求
1.2.3 socket()函式
對于C/C++,socket函式一般在WinSock2.h和WinSock.h庫中,開發者可通過在頭檔案中宣告
#include<WinSock2.h>
以引入WinSock2庫,
對于python,開發者可以通過
import socket
import sys
2.Python實作Socket通信
由于Python的Socket較容易理解,我們在這里使用Python入門Socket編程
2.1 Python中的Socket函式
Socket()格式如下
socket.socket([family[, type[, proto]]])
函式引數
- family: 套接字家族可以是 AF_UNIX 或者 AF_INET
- type: 套接字型別可以根據是面向連接的還是非連接分為
SOCK_STREAM或SOCK_DGRAM - protocol: 一般不填默認為0
2.2 Socket 物件(內建)方法
| 函式 | 描述 |
|---|---|
| 服務器端套接字 | |
| s.bind() | 系結地址(host,port)到套接字, 在AF_INET下,以元組(host,port)的形式表示地址, |
| s.listen() | 開始TCP監聽,backlog指定在拒絕連接之前,作業系統可以掛起的最大連接數量,該值至少為1,大部分應用程式設為5就可以了, |
| s.accept() | 被動接受TCP客戶端連接,(阻塞式)等待連接的到來 |
| 客戶端套接字 | |
| s.connect() | 主動初始化TCP服務器連接,,一般address的格式為元組(hostname,port),如果連接出錯,回傳socket.error錯誤, |
| s.connect_ex() | connect()函式的擴展版本,出錯時回傳出錯碼,而不是拋出例外 |
| 公共用途的套接字函式 | |
| s.recv() | 接收TCP資料,資料以字串形式回傳,bufsize指定要接收的最大資料量,flag提供有關訊息的其他資訊,通常可以忽略, |
| s.send() | 發送TCP資料,將string中的資料發送到連接的套接字,回傳值是要發送的位元組數量,該數量可能小于string的位元組大小, |
| s.sendall() | 完整發送TCP資料,完整發送TCP資料,將string中的資料發送到連接的套接字,但在回傳之前會嘗試發送所有資料,成功回傳None,失敗則拋出例外, |
| s.recvfrom() | 接收UDP資料,與recv()類似,但回傳值是(data,address),其中data是包含接收資料的字串,address是發送資料的套接字地址, |
| s.sendto() | 發送UDP資料,將資料發送到套接字,address是形式為(ipaddr,port)的元組,指定遠程地址,回傳值是發送的位元組數, |
| s.close() | 關閉套接字 |
| s.getpeername() | 回傳連接套接字的遠程地址,回傳值通常是元組(ipaddr,port), |
| s.getsockname() | 回傳套接字自己的地址,通常是一個元組(ipaddr,port) |
| s.setsockopt(level,optname,value) | 設定給定套接字選項的值, |
| s.getsockopt(level,optname[.buflen]) | 回傳套接字選項的值, |
| s.settimeout(timeout) | 設定套接字操作的超時期,timeout是一個浮點數,單位是秒,值為None表示沒有超時期,一般,超時期應該在剛創建套接字時設定,因為它們可能用于連接的操作(如connect()) |
| s.gettimeout() | 回傳當前超時期的值,單位是秒,如果沒有設定超時期,則回傳None, |
| s.fileno() | 回傳套接字的檔案描述符, |
| s.setblocking(flag) | 如果 flag 為 False,則將套接字設為非阻塞模式,否則將套接字設為阻塞模式(默認值),非阻塞模式下,如果呼叫 recv() 沒有發現任何資料,或 send() 呼叫無法立即發送資料,那么將引起 socket.error 例外, |
| s.makefile() | 創建一個與該套接字相關連的檔案 |
2.3 搭建服務端
我們使用 socket 模塊的 socket 函式來創建一個 socket 物件,socket 物件可以通過呼叫其他函式來設定一個 socket 服務,
現在我們可以通過呼叫 bind(hostname, port) 函式來指定服務的 port(埠),
接著,我們呼叫 socket 物件的 accept 方法,該方法等待客戶端的連接,并回傳 connection 物件,表示已連接到客戶端,
完整代碼如下
# server.py
# 匯入 socket、sys 模塊
import socket
import sys
# 創建 socket 物件
serversocket = socket.socket(
socket.AF_INET, socket.SOCK_STREAM)
# 獲取本地主機名
host = socket.gethostname()
port = 1234
# 系結埠號
serversocket.bind((host, port))
# 設定最大連接數,超過后排隊
serversocket.listen(5)
while True:
# 建立客戶端連接
clientsocket,addr = serversocket.accept()
print("連接地址: %s" % str(addr))
msg='connect successfully!"
# 向客戶端發送資料
clientsocket.send(msg.encode('utf-8'))
clientsocket.close()
2.4 搭建客戶端
接下來我們寫一個簡單的客戶端實體連接到以上創建的服務,埠號為 1234,
socket.connect(hosname, port ) 方法打開一個 TCP 連接到主機為 hostname 埠為 port 的服務商,連接后我們就可以從服務端獲取資料,記住,操作完成后需要關閉連接,
完整代碼如下
# client.py
# 匯入 socket、sys 模塊
import socket
import sys
# 創建 socket 物件
s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
# 獲取本地主機名
host = socket.gethostname()
# 設定埠號
port = 1234
# 連接服務,指定主機和埠
s.connect((host, port))
# 接收小于 1024 位元組的資料
msg = s.recv(1024)
s.close()
print (msg.decode('utf-8'))
2.5 運行
我們現在先運行服務端,再運行客戶端
在客戶端的程式里會有下面的輸出
$ python3 client.py
connect successfully!
同時,在服務端會有一下輸出
連接地址: ('你的Ip地址', 客戶端的埠)
轉載請註明出處,本文鏈接:https://www.uj5u.com/ruanti/293556.html
標籤:其他
