您好,登錄后才能下訂單哦!
一.什么是socket?
socket就是為了實現(xiàn)C/S架構而生的,socket位于應用層和傳輸層之間,是傳輸層和應用層之間的一組接口,它把復雜的TCP/IP協(xié)議族隱藏在Socket接口后面,對用戶來說,一組簡單的接口就是全部,讓Socket去組織數(shù)據(jù),以符合指定的協(xié)議,所以,我們無需深入理解tcp/udp協(xié)議,socket已經為我們封裝好了,我們只需要遵循socket的規(guī)定去編程,寫出的程序自然就是遵循tcp/udp標準的。
也有人將socket說成ip+port,ip是用來標識互聯(lián)網中的一臺主機的位置,而port是用來標識這臺機器上的一個應用程序,ip地址是配置到網卡上的,而port是應用程序開啟的,ip與port的綁定就標識了互聯(lián)網中獨一無二的一個應用程序。
補充:有人會把socket和程序的pid弄混,程序的pid是同一臺機器上不同進程或者線程的標識。
二.socket種類。
基于文件型套接字。
AF_UNIX:基于文件型的套接字,就是通過底層的文件系統(tǒng)來取數(shù)據(jù),兩個套接字進程運行在同一臺機器,可以通過訪問同一個文件系統(tǒng)來實現(xiàn)兩個程序之間的通信。
基于網絡型套接字。
三.socket工作流程。
tcp服務端建立連接流程:
socket()實例化出一個套接字對象→bind()綁定ip地址和端口→listen()開始監(jiān)聽之前綁定的ip地址和端口→accept()開始被動接收tcp客戶端發(fā)來的連接,會一直等連接的到來(如果沒有連接,就會一直等,直到有客戶端連過來。
tcp客戶端建立連接流程:
socket()客戶端也初始化一個套接字對象→connect()主動去連接服務端(主動初始化與tcp服務器的連接)如果連接成功,客戶端和服務器之間就可以互相收發(fā)數(shù)據(jù)了。(其實connect就相當于去回應服務端的accept,服務端的accept一直在等待客戶端去connect)
客戶端發(fā)送數(shù)據(jù)請求,服務器端接收請求并處理請求,然后把回應數(shù)據(jù)發(fā)送給客戶端,客戶端讀取數(shù)據(jù),最后關閉連接,一次交互結束。
四.socket模塊的用法。
import socket
socket.socket(socket_family,socket_type,protocal=0)
socket_family 可以是 AF_UNIX 或 AF_INET。socket_type 可以是 SOCK_STREAM 或 SOCK_DGRAM。protocol 一般不填,默認值為 0。
獲取tcp/ip套接字
tcpSock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
獲取udp/ip套接字
udpSock = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
由于 socket 模塊中有太多的屬性。我們在這里破例使用了'from module import *'語句。使用 'from socket import *',我們就把 socket 模塊里的所有屬性都帶到我們的命名空間里了,這樣能 大幅減短我們的代碼。
例如tcpSock = socket(AF_INET, SOCK_STREAM)
服務端常用專屬套接字方法:
s.bind():綁定(主機,端口號)到套接字。
s.listen():開始進行tcp監(jiān)聽。#listen方法中可以指定一個backlog參數(shù),這個參數(shù)用于設置tcp連接池的大?。P于tcp連接池和三次握手的知識后面會做補充。)
s.accept():被動接受TCP客戶的連接,(阻塞式)等待連接的到來。
客戶端常用專屬套接字方法:
s.connect():主動初始化TCP服務器連接
s.connect_ex() : connect()函數(shù)的擴展版本,出錯時返回出錯碼,而不是拋出異常。
客戶端和服務端共同具有的常用套接字方法:
s.recv():接收tcp數(shù)據(jù)。#需要指定每次收多少字節(jié)。
s.send(): 發(fā)送tcp數(shù)據(jù),#發(fā)送TCP數(shù)據(jù)(send在待發(fā)送數(shù)據(jù)量大于對端緩存區(qū)剩余空間時,數(shù)據(jù)丟失,不會發(fā)完)
s.sendall(): 發(fā)送tcp數(shù)據(jù),發(fā)送完整的TCP數(shù)據(jù)(本質就是循環(huán)調用send,sendall在待發(fā)送數(shù)據(jù)量大于對端緩存區(qū)剩余空間時,數(shù)據(jù)不丟失,循環(huán)調用send直到發(fā)完)(個人分析,本質上是調用了循環(huán))
s.recvfrom():接收UDP數(shù)據(jù)
s.sendto() : 發(fā)送UDP數(shù)據(jù)
s.getpeername() : 連接到當前套接字的遠端的地址
s.getsockname() : 當前套接字的地址
s.getsockopt() :返回指定套接字的參數(shù)
s.setsockopt() : 設置指定套接字的參數(shù)
s.close() :關閉套接字
五.socket用法示例。
服務端:
#!/usr/bin/python2.7
# -*- coding:utf-8 -*-
import socket
address_and_port = ('127.0.0.1',8888)
s1 = socket.socket(family=socket.AF_INET,type=socket.SOCK_STREAM)
s1.bind(address_and_port)
s1.listen(3)
conn,addr = s1.accept() #執(zhí)行了socket的accept方法后,開始阻塞,等待客戶端的連接.
#執(zhí)行了accept方法后,會返回一個元祖,這個元祖里面包含了一個客戶端的連接對象,還有客戶端的ip地址和端口.
while True:
data = conn.recv(1024) #用于接收tcp數(shù)據(jù),后面的1024表示,recv一次最多可以接收1024字節(jié).(執(zhí)行到recv,如果對端沒有發(fā)來數(shù)據(jù),或者內容為空,程序會阻塞住,不繼續(xù)向下執(zhí)行。)
print data
conn.send(data.upper())
conn.close() #關閉與客戶端的連接
s1.close() #關閉服務端套接字連接
客戶端:
#!/usr/bin/python2.7
# -*- coding:utf-8 -*-
import socket
s1 = socket.socket(socket.AF_INET,socket.SOCK_STREAM)
s1.connect(('127.0.0.1',8888))
while True:
msg = raw_input(">>>").strip()
s1.send(msg.encode(('utf-8')))
print "客戶端已發(fā)送數(shù)據(jù)!"
data = s1.recv(1024)
print data
#上面兩個代碼,可以簡單理解下socket的工作流程。
六.解決sockt 通信客戶端斷開服務端崩潰等問題的解決方法(連接循環(huán),通信循環(huán),以及異常捕捉)
服務端:
#!/usr/bin/python2.7
# -*- coding:utf-8 -*-
import socket
address_and_port = ('127.0.0.1',8888)
s1 = socket.socket(family=socket.AF_INET,type=socket.SOCK_STREAM)
s1.bind(address_and_port)
s1.listen(3)
while True: #此處問連接循環(huán),會循環(huán)等連接。
conn,addr = s1.accept() #執(zhí)行了socket的accept方法后,開始阻塞,等待客戶端的連接.
#執(zhí)行了accept方法后,會返回一個元祖,這個元祖里面包含了一個客戶端的連接對象,還有客戶端的ip地址和端口.
while True: #數(shù)據(jù)傳輸循環(huán)
try: #防止客戶端斷開導致的服務斷拋異常。
data = conn.recv(1024) #用于接收tcp數(shù)據(jù),后面的1024表示,recv一次最多可以接收1024字節(jié).
if not data:
break
print data
conn.send(data.upper())
except Exception:
break
conn.close() #關閉與客戶端的連接
s1.close() #關閉服務端套接字連接
客戶端:
#!/usr/bin/python2.7
# -*- coding:utf-8 -*-
import socket
s1 = socket.socket(socket.AF_INET,socket.SOCK_STREAM)
s1.connect(('127.0.0.1',8888))
while True:
msg = raw_input(">>>").strip()
if not msg: #防止客戶端發(fā)送空數(shù)據(jù)
continue
s1.send(msg.encode(('utf-8')))
print "客戶端已發(fā)送數(shù)據(jù)!"
data = s1.recv(1024)
print data
s1.close()
#以上是防止服務端崩潰的一些解決方法。
七.關于udp套接字。
服務端大概操作流程:
ss = socket() #創(chuàng)建一個服務器的套接字
ss.bind() #綁定服務器套接字
inf_loop: #服務器無限循環(huán)
cs = ss.recvfrom()/ss.sendto() # 對話(接收與發(fā)送)
ss.close() # 關閉服務器套接字
客戶端大概的操作流程:
cs = socket() # 創(chuàng)建客戶套接字
comm_loop: # 通訊循環(huán)
cs.sendto()/cs.recvfrom() # 對話(發(fā)送/接收)
cs.close() # 關閉客戶套接字
下面是udp套接字的一個簡單的示例:
示例1:
服務端:
import socket
udp_server = socket.socket(socket.AF_INET,socket.SOCK_DGRAM)
udp_server.bind(('127.0.0.1',8888))
while True:
msg,addr = udp_server.recvfrom(1024)
print msg ,addr
udp_server.sendto(msg.upper(),addr)
客戶端:
import socket
udp_client = socket.socket(socket.AF_INET,socket.SOCK_DGRAM)
while True:
msg = raw_input(">>:").strip()
if not msg:
continue
udp_client.sendto(msg.encode('utf-8'),("127.0.0.1",8888))
back_msg,addr = udp_client.recvfrom(1024)
print back_msg.decode('utf-8') ,addr
八.關于tcp套接字和udp套接字你需要知道的。
首先有幾點必須明確!
1.1 所有的收發(fā)消息,都是在操作自己的tcp/udp緩沖區(qū),發(fā)消息是將數(shù)據(jù)發(fā)送到自己的緩沖區(qū)中,收消息同樣也是從緩沖區(qū)中收取。
1.2 tcp 使用send發(fā)消息,使用recv收消息
1.3udp 使用sendto發(fā)消息,使用recvfrom收消息
tcp與udp
2.1 tcp是基于流的,udp是基于報的
2.2 使用send發(fā)送數(shù)據(jù)流的時候,若數(shù)據(jù)流為空,那么tcp buffer(緩沖區(qū))也為空,操作系統(tǒng)不會控制tcp發(fā)包。
sendinto(bytes_data,ip_port):發(fā)送數(shù)據(jù)報,bytes_data為空,還有ip_port,所有即便是發(fā)送空的bytes_data,數(shù)據(jù)報其實也不是空的,自己這端的緩沖區(qū)收到內容,操作系統(tǒng)就會控制udp協(xié)議發(fā)包。
3.關于tcp的補充
tcp基于鏈接通信:
基于鏈接,則需要listen(backlog),指定半連接池的大小
基于鏈接,必須先運行的服務端,然后客戶端發(fā)起鏈接請求
對于mac系統(tǒng):如果一端斷開了鏈接,那另外一端的鏈接也跟著完蛋recv將不會阻塞,收到的是空(解 決方法是:服務端在收消息后加上if判斷,空消息就break掉通信循環(huán))
對于windows/linux系統(tǒng):如果一端斷開了鏈接,那另外一端的鏈接也跟著完蛋recv將不會阻塞,收到 的是空(解決方法是:服務端通信循環(huán)內加異常處理,捕捉到異常后就break掉通訊循環(huán))
udp通信。
無鏈接,因而無需listen(backlog),更加沒有什么連接池之說了
無鏈接,udp的sendinto不用管是否有一個正在運行的服務端,可以己端一個勁的發(fā)消息,只不過數(shù)據(jù)丟失
recvfrom收的數(shù)據(jù)小于sendinto發(fā)送的數(shù)據(jù)時,在mac和linux系統(tǒng)上數(shù)據(jù)直接丟失,在windows系統(tǒng)上發(fā)送的比接收的大直接報錯
只有sendinto發(fā)送數(shù)據(jù)沒有recvfrom收數(shù)據(jù),數(shù)據(jù)丟失。
注意:
1.你單獨運行上面的udp的客戶端,你發(fā)現(xiàn)并不會報錯,相反tcp卻會報錯,因為udp協(xié)議只負責把包發(fā)出去,對方收不收,我根本不管,而tcp是基于鏈接的,必須有一個服務端先運行著,客戶端去跟服務端建立鏈接然后依托于鏈接才能傳遞消息,任何一方試圖把鏈接摧毀都會導致對方程序的崩潰。
2.上面的udp程序,你注釋任何一條客戶端的sendinto,服務端都會卡住,為什么?因為服務端有幾個recvfrom就要對應幾個sendinto,哪怕是sendinto(b'')那也要有。
免責聲明:本站發(fā)布的內容(圖片、視頻和文字)以原創(chuàng)、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關證據(jù),一經查實,將立刻刪除涉嫌侵權內容。