Socket的英文原義是「孔」或「插座」。作為BSD UNIX的進程通訊機制,通常也稱為"套接字",用於描述IP位址和端口,是一個通訊鏈的句柄,可以用來實現不同虛擬機或不同電腦之間的通訊。
網路上的兩個程式透過一個雙向的通訊連線實現資料的交換,這個連線的一端稱為一個socket。
建立網路通訊連線至少要一對埠號(socket)。 socket本質是程式介面(API),對TCP/IP的封裝,TCP/IP也要提供可供程式設計師做網路開發所用的接口,這就是Socket程式介面;HTTP是轎車,提供了封裝或顯示資料的具體形式;Socket是發動機,提供了網路通訊的能力。
下面來說一下python的socket。
要使用socket.socket()函數來建立套接字。其語法如下:
socket.socket(socket_family,socket_type,protocol=0)
socket_family可以是以下參數:
socket.AF_INET-#4(預設)
# socket.AF_INET6 IPv6 socket.AF_UNIX 只能夠用於單一的Unix系統進程間通訊socket_type可以是以下參數: 流式串流. socket , for TCP (預設) socket.SOCK_DGRAM 資料報式socket , for UDP socket.SOCK_RAW 原始套接字,普通的套接字等網路封包文,而SOCK_RAW可以;其次,SOCK_RAW也可以處理特殊的IPv4報文;此外,利用原始套接字,可以透過IP_HDRINCL套接字選項由使用者建構IP頭。 socket.SOCK_RDM 是一種可靠的UDP形式,即保證交付資料封包但不保證順序。 SOCK_RAM用來提供對原始協定的低階訪問,在需要執行某些特殊操作時使用,例如發送ICMP封包。 SOCK_RAM通常僅限於高級使用者或管理員執行的程式使用。 socket.SOCK_SEQPACKET 可靠的連續資料包服務protocol參數: 0 (預設)與特定的位址家族相關的協定,如果是0 ,則系統就會根據位址格式和套接類別,自動選擇一個適當的協定2.套接字物件內建方法伺服器端套接字函數#s. bind() 綁定位址(ip位址,連接埠)到套接字,參數必須是元組的格式例如:s.bind(('127.0.0.1',8009))#s.listen( 5) 開始監聽,5為最大掛起的連接數s.accept() 被動接受客戶端連接,阻塞,等待連接客戶端套接字函數s.connect() 連接伺服器端,參數必須是元組格式例如:s.connect(('127,0.0.1',8009))#公共用途的套接字函數s.recv(1024) 接收TCP數據,1024為一次數據接收的大小s.send(bytes) 發送TCP數據,python3發送數據的格式必須為bytes格式s.sendall() 完整傳送數據,內部循環呼叫sends.close() 關閉套接字實例1.簡單實作socket程式#server端#!/usr/bin/env python # _*_ coding:utf-8 _*_ import socket import time IP_PORT = ('127.0.0.1',8009) BUF_SIZE = 1024 tcp_server = socket.socket() tcp_server.bind(IP_PORT) tcp_server.listen(5) while True: print("waiting for connection...") conn,addr = tcp_server.accept() print("...connected from:",addr) while True: data = tcp_server.recv(BUF_SIZE) if not data:break tcp_server.send('[%s] %s'%(time.ctime(),data)) tcp_server.close()
#!/usr/bin/env python # _*_ coding:utf-8 _*_ import socket HOST = '127.0.0.1' PORT = 8009 BUF_SIZE = 1024 ADDR = (HOST,PORT) client = socket.socket() client.connect(ADDR) while True: data = input(">>> ") if not data:break client.send(bytes(data,encoding='utf-8')) recv_data = client.recv(BUF_SIZE) if not recv_data:break print(recv_data.decode()) client.close()
[root@pythontab]# python client.py >>> hello python [Thu Sep 15 22:29:12 2016] b'hello python'
[root@pythontab]# python server.py waiting for connection... ...connected from: ('127.0.0.1', 55378)
服务端程序代码
#!/usr/bin/env python # _*_ coding:utf-8 _*_ import socketserver import time HOST = '127.0.0.1' PORT = 8009 ADDR = (HOST,PORT) BUF_SIZE = 1024 class Myserver(socketserver.BaseRequestHandler): def handle(self): while True: print("...connected from:",self.client_address) data = self.request.recv(BUF_SIZE) if not data:break self.request.send(bytes("%s %s"%(time.ctime(),data))) server = socketserver.ThreadingTCPServer(ADDR,Myserver) print("waiting for connection...") server.serve_forever()
11~17行
主要的工作在这里。从socketserver的BaseRequestHandler类中派生出一个子类,并重写handle()函数。
在有客户端发进来的消息的时候,handle()函数就会被调用。
19~21行
代码的最后一部分用给定的IP地址和端口加上自定义处理请求的类(Myserver)。然后进入等待客户端请求与处理客户端请求的无限循环中。
客户端程序代码
import socket HOST = '127.0.0.1' PORT = 8009 ADDR = (HOST,PORT) BUF_SIZE = 1024 client = socket.socket() client.connect(ADDR) while True: data = input(">>> ") if not data:continue client.send(bytes(data,encoding='utf-8')) recv_data = client.recv(BUF_SIZE) if not recv_data:break print(recv_data.decode()) client.close()
执行服务端和客户端代码
下面是客户端输出
[root@pythontab]# python socketclient.py >>> hello python Thu Sep 15 23:53:31 2016 b'hello python' >>> hello pythontab Thu Sep 15 23:53:49 2016 b'hello pythontab'
下面是服务端输出
[root@pythontab]# python socketserver.py waiting for connection... ...connected from: ('127.0.0.1', 55385) ...connected from: ('127.0.0.1', 55385) ...connected from: ('127.0.0.1', 55385) ...connected from: ('127.0.0.1', 55385)
以上是python中socket模組詳解的詳細內容。更多資訊請關注PHP中文網其他相關文章!