一 簡介
socket是兩個應用程序進行通信的管道,這兩個應用程序可以在同一台機器上,也可以位於兩台不同的機器上,相同的網絡或者不同網絡之間的。Python socket有豐富的類和方法,可以簡化socket編程。本文算是一個學習筆記。
二 socket
2.1 socket類型
socket.AF_UNIX
只能夠用於單一的Unix系統進程間通信
socket.AF_INET
服務器之間網絡通信
socket.AF_INET6
用於IPv6網絡的通信
socket.SOCK_STREAM
流式socket , for TCP
socket.SOCK_DGRAM
數據報式socket , for UDP
socket.SOCK_RAW
原始套接字,普通的套接字無法處理ICMP、IGMP等網絡報文,而SOCK_RAW可以;其次,SOCK_RAW也可以處理特殊的IPv4報文;此外,利用原始套接字,可以通過IP_HDRINCL套接字選項由用戶構造IP頭。
socket.SOCK_SEQPACKET
可靠的連續數據包服務
例子
創建TCP Socket:
import socket
tcpSerSock=socket.socket(socket.AF_INET,socket.SOCK_STREAM)
創建UDP Socket:
tcpSerSock=socket.socket(socket.AF_INET,socket.SOCK_DGRAM)
注意點:
1)TCP發送數據時,已建立好TCP連接,所以不需要指定地址。UDP是面向無連接的,每次發送要指定是發給誰。
2)服務端與客戶端不能直接發送列表,元組,字典。需要字符串化repr(data)。
2.2 socket函數
服務端socket函數
socket.bind(address)
將套接字綁定到地址, 在AF_INET下,以元組(host,port)的形式表示地址.
socket.listen(backlog)
開始監聽TCP傳入連接。backlog指定在拒絕連接之前,操作系統可以掛起的最大連接數量。該值至少為1。
socket.accept()
接受TCP連接並返回(conn,address),其中conn是新的套接字對象,可以用來接收和發送數據。address是連接客戶端的地址。
客戶端socket函數
socket.connect(address)
連接到address處的套接字。一般address的格式為元組(hostname,port),如果連接出錯,返回socket.error錯誤。
socket.connect_ex(adddress)
功能與connect(address)相同,但是成功返回0,失敗返回errno的值。
公用函數
socket.recv(bufsize[,flag])
接受TCP套接字的數據。數據以字符串形式返回,bufsize指定要接收的最大數據量。flag提供有關消息的其他信息,通常可以忽略。
socket.send(string[,flag])
發送TCP數據。將string中的數據發送到連接的套接字。返回值是要發送的字節數量,該數量可能小於string的字節大小。
socket.sendall(string[,flag])
完整發送TCP數據。將string中的數據發送到連接的套接字,但在返回之前會嘗試發送所有數據。成功返回None,失敗則拋出異常。
socket.recvfrom(bufsize[.flag])
接受UDP套接字的數據。與recv()類似,但返回值是(data,address)。其中data是包含接收數據的字符串,address是發送數據的套接字地址。
socket.sendto(string[,flag],address)
發送UDP數據。將數據發送到套接字,address是形式為(ipaddr,port)的元組,指定遠程地址。返回值是發送的字節數。
socket.close()
關閉套接字。
socket.getpeername()
返回連接套接字的遠程地址。返回值通常是元組(ipaddr,port)。
socket.getsockname()
返回套接字自己的地址。通常是一個元組(ipaddr,port)
socket.setsockopt(level,optname,value)
設置給定套接字選項的值。
socket.getsockopt(level,optname[.buflen])
返回套接字選項的值。
socket.settimeout(timeout)
設置套接字操作的超時期,timeout是一個浮點數,單位是秒。值為None表���沒有超時期。一般,超時期應該在剛創建套接字時設置,因為它們可能用於連接的操作(如connect())
socket.fileno()
返回套接字的文件描述符。
socket.setblocking(flag)
如果flag為0,則將套接字設為非阻塞模式,否則將套接字設為阻塞模式(默認值)。非阻塞模式下,如果調用recv
socket.makefile()
創建一個與該套接字相關連的文件
三 創建socket連接
服務端
1 創建socket對象。
tcpSock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
2 使用bind方法來將socket綁定到指定地址和端口。
tcpSock.bind(address,port)
3 使用listen方法接收連接請求。
tcpSock.listen( backlog )
backlog指定最多允許多少個客戶連接到服務器。它的值至少為1。收到連接請求後,這些請求需要排隊,如果隊列滿,就拒絕請求。
4 通過socket的accept方法等待客戶請求連接。
connection, address = socket.accept()
調用accept方法時,socket會時入"waiting"狀態,等待客戶請求連接,當客戶端發送請求到服務端時,accept方法建立連接並返回服務器。accept方法返回一個含有兩個元素的 元組(connection,address)。
第一個元素 connection是新的socket對象,服務器必須通過它與客戶通信;
第二個元素 address是客戶的IP地址。
5 處理階段,服務器和客戶端通過send和recv方法通信(傳輸數據)。服務器調用send方法以字符串形式向客戶發送數據,也可以使用recv方法從客戶接收信息。調用recv時,
服務器必須設置接收的最大數據量。recv方法在接收數據時會進入“blocked”狀態,最後返回一個字符串,用它表示收到的數據。
如果發送的數據量超過recv所允許接收的限制,數據會被截斷。多余的數據將緩沖於接收端。下一次調用recv時,多余的數據會從緩沖區刪除(以及自上次調用recv以來,客戶可能發送的其它任何數據)
6 傳輸結束,服務器調用socket的close方法關閉連接。
客戶端
1 創建一個socket以連接服務器,和服務器端創建socket對象一樣。
tcpCliSock = socket.socket(socket.AF_INET,socket.SOCK_STREAM)
2 使用socket的connect方法連接服務器。
tcpCliSock.connect( (host,port) )
host代表服務器主機名或IP,port代表服務器進程所綁定的端口號。如連接成功,客戶就可通過套接字與服務器通信,如果連接失敗,會引發socket.error異常。
3 處理階段,客戶和服務器將通過send方法和recv方法通信。
tcpCliSock.send(data)
4 傳輸結束,客戶通過調用socket的close方法關閉連接。
tcpCliSock.close()
一圖勝千言 ,socket 建立連接的過程 via
更多詳情見請繼續閱讀下一頁的精彩內容: http://www.linuxidc.com/Linux/2016-04/130540p2.htm