一.內容回顧
socket通常也稱作"套接字",用于描述IP地址和端口,是一個通信鏈的句柄,應用程序通常通過"套接字"向網絡發出請求或者應答網絡請求。
socket起源于Unix,而Unix/Linux基本哲學之一就是“一切皆文件”,對于文件用【打開】【讀寫】【關閉】模式來操作。socket就是該模式的一個實現,socket即是一種特殊的文件,一些socket函數就是對其進行的操作(讀/寫IO、打開、關閉)
socket和file的區別:
- file模塊是針對某個指定文件進行【打開】【讀寫】【關閉】
- socket模塊是針對 服務器端 和 客戶端Socket 進行【打開】【讀寫】【關閉】
#tcp協議#三次握手 建立連接#數據的傳遞 有連接的\全雙工#效率低可靠#四次揮手 斷開連接 #udp協議# 無連接的\快\不可靠的
# tcp協議打交道 # 黏包現象 # 解決黏包現象# 并發編程 # 同一時刻只能和一個客戶端通信 # 解決占線問題
?tcp協議格式
sk = socket.socket()#sk = socket.socket(socket.AF_INET,socket.SOCK_STREAM,0) 參數一:地址簇socket.AF_INET IPv4(默認)socket.AF_INET6 IPv6socket.AF_UNIX 只能夠用于單一的Unix系統進程間通信參數二:類型socket.SOCK_STREAM 流式socket , for TCP (默認)socket.SOCK_DGRAM 數據報式socket , for UDPsocket.SOCK_RAW 原始套接字,普通的套接字無法處理ICMP、IGMP等網絡報文,而SOCK_RAW可以;其次,SOCK_RAW也可以處理特殊的IPv4報文;此外,利用原始套接字,可以通過IP_HDRINCL套接字選項由用戶構造IP頭。socket.SOCK_RDM 是一種可靠的UDP形式,即保證交付數據報但不保證順序。SOCK_RAM用來提供對原始協議的低級訪問,在需要執行某些特殊操作時使用,如發送ICMP報文。SOCK_RAM通常僅限于高級用戶或管理員運行的程序使用。socket.SOCK_SEQPACKET 可靠的連續數據包服務參數三:協議0 (默認)與特定的地址家族相關的協議,如果是 0 ,則系統就會根據地址格式和套接類別,自動選擇一個合適的協議
addr = ('127.0.0.1',9000)#sever的地址 sk.bind(addr) sk.bind(address) s.bind(address) 將套接字綁定到地址。address地址的格式取決于地址族。在AF_INET下,以元組(host,port)的形式表示地址。
sk.listen(backlog)
開始監聽傳入連接。backlog指定在拒絕連接之前,可以掛起的最大連接數量。backlog等于5,表示內核已經接到了連接請求,但服務器還沒有調用accept進行處理的連接個數最大為5這個值不能無限大,因為要在內核中維護連接隊列
sk.accept()
接受連接并返回(conn,address),其中conn是新的套接字對象,可以用來接收和發送數據。address是連接客戶端的地址。接收TCP 客戶的連接(阻塞式)等待連接的到來
sk.recv(bufsize[,flag])
接受套接字的數據。數據以字符串形式返回,bufsize指定最多可以接收的數量。flag提供有關消息的其他信息,通常可以忽略
sk.recvfrom(bufsize[.flag])與recv()類似,但返回值是(data,address)。其中data是包含接收數據的字符串,address是發送數據的套接字地址。
sk.send(string[,flag])
將string中的數據發送到連接的套接字。返回值是要發送的字節數量,該數量可能小于string的字節大小。
sk.sendall(string[,flag])
將string中的數據發送到連接的套接字,但在返回之前會嘗試發送所有數據。成功返回None,失敗則拋出異常。
sk.sendto(string[,flag],address)
***將數據發送到套接字,address是形式為(ipaddr,port)的元組,指定遠程地址。返回值是發送的字節數。該函數主要用于UDP協議。
sk.settimeout(timeout)
設置套接字操作的超時期,timeout是一個浮點數,單位是秒。值為None表示沒有超時期。一般,超時期應該在剛創建套接字時設置,因為它們可能用于連接的操作(如 client 連接最多等待5s )
sk.getpeername()
返回連接套接字的遠程地址。返回值通常是元組(ipaddr,port)。
sk.getsockname()
返回套接字自己的地址。通常是一個元組(ipaddr,port)
sk.fileno()
套接字的文件描述符
tcp 與 udp
tcp協議有sk.listen()這個語句,udp沒有
tcp 發送sever 發送消息用send,也可以用sendto,conn.send/sk.send
udp發送消息用sendto
udp里sk = socket.socket(type = socket.SOCK_DGRAM)
inp = {'filename': 'timg','filesize':59116} msg = str(inp).encode('utf-8') ret = struct.pack('i',len(msg)) conn.send(ret)
n = sk.recv(4) ret = struct.unpack('i',n)
struct.pack
struct.pack用于將Python的值根據格式符,轉換為字符串(因為Python中沒有字節(Byte)類型,可以把這里的字符串理解為字節流,或字節數組)。其函數原型為:struct.pack(fmt, v1, v2, …),參數fmt是格式字符串,關于格式字符串的相關信息在下面有所介紹。v1, v2, …表示要轉換的python值。格式符”i”表示轉換為int,’ii’表示有兩個int變量。
進行轉換后的結果長度為8個字節(int類型占用4個字節,兩個int為8個字節)
可以看到輸出的結果是亂碼,因為結果是二進制數據,所以顯示為亂碼。
可以使用python的內置函數repr來獲取可識別的字符串,其中十六進制的0x00000014, 0x00001009分別表示20和400。struct.unpack
struct.unpack做的工作剛好與struct.pack相反,用于將字節流轉換成python數據類型。它的函數原型為:struct.unpack(fmt, string),該函數返回一個元組。
?并發:
import socketserverclass MyServer(socketserver.BaseRequestHandler):def handle(self):# 這個handle方法是每有一個客戶端發起connect之后,就會執行handle# 在建立連接之后的所有內容都在handle中實現就可以了# ThreadingTCPServer幫助我們完成了tcp協議的server端的并發conn = self.requestwhile True:msg = conn.recv(1024).decode('utf-8')print(msg)conn.send(msg.upper().encode('utf-8'))server = socketserver.ThreadingTCPServer(('127.0.0.1',9000),MyServer) server.serve_forever()
?