溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

10.python網絡編程(startin part 1)

發(fā)布時間:2020-04-07 15:46:56 來源:網絡 閱讀:930 作者:蘇浩智 欄目:建站服務器

一.什么是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種類。

  1. 基于文件型套接字。

    AF_UNIX:基于文件型的套接字,就是通過底層的文件系統(tǒng)來取數(shù)據(jù),兩個套接字進程運行在同一臺機器,可以通過訪問同一個文件系統(tǒng)來實現(xiàn)兩個程序之間的通信。

  2. 基于網絡型套接字。

三.socket工作流程。

10.python網絡編程(startin part 1)

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.1  所有的收發(fā)消息,都是在操作自己的tcp/udp緩沖區(qū),發(fā)消息是將數(shù)據(jù)發(fā)送到自己的緩沖區(qū)中,收消息同樣也是從緩沖區(qū)中收取。

    1.2 tcp 使用send發(fā)消息,使用recv收消息

    1.3udp 使用sendto發(fā)消息,使用recvfrom收消息

  2. 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'')那也要有。


向AI問一下細節(jié)

免責聲明:本站發(fā)布的內容(圖片、視頻和文字)以原創(chuàng)、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關證據(jù),一經查實,將立刻刪除涉嫌侵權內容。

AI