標籤:.com 位元組 pre 解決方案 http 執行個體化 拒絕 錯誤 continue
socket
socket通常也稱作"通訊端",用於描述IP地址和連接埠,是一個通訊鏈的控制代碼,應用程式通常通過"通訊端"向網路發出請求或者應答網路請求。
sk.bind(address)
s.bind(address) 將通訊端綁定到地址。address地址的格式取決於地址族。在AF_INET下,以元組(host,port)的形式表示地址。
sk.listen(backlog)
開始監聽傳入串連。backlog指定在拒絕串連之前,可以掛起的最大串連數量。
backlog等於5,表示核心已經接到了串連請求,但伺服器還沒有調用accept進行處理的串連個數最大為5
這個值不能無限大,因為要在核心中維護串連隊列
sk.setblocking(bool)
是否阻塞(預設True),如果設定False,那麼accept和recv時一旦無資料,則報錯。
sk.accept()
接受串連並返回(conn,address),其中conn是新的通訊端對象,可以用來接收和發送資料。address是串連用戶端的地址。
接收TCP 客戶的串連(阻塞式)等待串連的到來
sk.connect(address)
串連到address處的通訊端。一般,address的格式為元組(hostname,port),如果串連出錯,返回socket.error錯誤。
sk.connect_ex(address)
同上,只不過會有傳回值,串連成功時返回 0 ,串連失敗時候返回編碼,例如:10061
sk.close()
關閉通訊端
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,失敗則拋出異常。
內部通過遞迴調用send,將所有內容發送出去。
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()
通訊端的檔案描述符
import socket server = socket.socket() # 產生socket控制代碼# server = socket.socket(AF.INET, sock_STREAM)server.bind((‘localhost‘, 6960)) # 綁定監控的連接埠server.listen(5) # 監聽綁定的連接埠print(‘waitting...‘) conn, addr = server.accept() # 等待用戶端串連print(conn, addr) while True: data = conn.recv(1024) # 接收用戶端資料 print(‘recv:‘, data.decode()) conn.send(data) # 向用戶端發送資料 conn.close()
伺服器端
import socket client = socket.socket() # 產生socket控制代碼client.connect((‘localhost‘, 9999)) # 連結的伺服器端 while True: msg = input(‘>>‘).strip() if msg == 0: continue client.send(msg.encode(‘utf-8‘)) # 向伺服器端發送資料 data = client.recv(1024) print(‘recv:‘, data.decode()) # 接收伺服器端資料 client.close()
用戶端
存在的問題
1.用戶端斷開時,伺服器端也會自動斷開報錯。
解決方案:a.添加異常處理機制。
b.伺服器端判斷接受到的字串是否為空白(用戶端斷開後,會繼續發送Null 字元)。
c.伺服器使用socketserver。
2.可能存在的粘包問題(send函數發送資料,有一種機制會當發送的資料達到緩衝區大小時或經過一個接受時延,再統一發送,而不是send一次發送一次)。
解決方案:a.每發送一個檔案,接受方收到後返回一個確實收到值,避免連續發送。
b.傳送檔案前,先將檔案大小發送給接收方,使接受方只接收相應大小的資料。
實現簡單的ssh
import socket,os#執行個體化socketser=socket.socket()#綁定ip與連接埠ser.bind((‘localhost‘,9999))#監聽ser.listen()while True: #阻塞,等待連結 print("開始等!") conn,addr=ser.accept() while True: #接受命令 try: data=conn.recv(1024) #執行命令返回字元 print("開始執行命令!") cmd_res=os.popen(data.decode()).read() if len(cmd_res)==0: cmd_res="命令錯誤" #發送字元長度 conn.send(str(len(cmd_res.encode("utf-8"))).encode("utf-8")) #接受回應,防止粘包 conn.recv(1024) #發送命令結果 conn.send(cmd_res.encode("utf-8")) print("發送完成!") except Exception as e: print(e) break伺服器端
import socket#聲明socket對象cli=socket.socket()#連結cli.connect(("localhost",9999))while True: #輸入命令 cmd=input(">>:") #傳送命令 cli.send(cmd.encode("utf-8")) data_size=int(cli.recv(1024).decode()) cli.send(b"200") recv_data=b"" recv_size=0 #迴圈接受資料直到收完 while data_size>recv_size: #防止粘包 if data_size-recv_size<1024: size=data_size-recv_size else: size=1024 #接受資料 data=cli.recv(size) recv_size+=len(data) recv_data+=data else: print(recv_data.decode())用戶端
攻克python-socket