溫馨提示×

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

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

python怎么使用socket高效傳輸視頻數(shù)據(jù)幀

發(fā)布時(shí)間:2021-10-23 17:01:14 來源:億速云 閱讀:343 作者:iii 欄目:開發(fā)技術(shù)

本篇內(nèi)容主要講解“python怎么使用socket高效傳輸視頻數(shù)據(jù)幀”,感興趣的朋友不妨來看看。本文介紹的方法操作簡(jiǎn)單快捷,實(shí)用性強(qiáng)。下面就讓小編來帶大家學(xué)習(xí)“python怎么使用socket高效傳輸視頻數(shù)據(jù)幀”吧!

目錄
  • 遇到的問題

  • 代碼問題記錄(需要代碼的可以直接文末)

  • 代碼

    • 客戶端clien.py

    • 服務(wù)端server.py

遇到的問題

網(wǎng)上找了一些代碼,都是只能建立一次socket傳輸一張圖片,然后斷開重新連重新傳。而建立一次socket代價(jià)不小,反復(fù)建立會(huì)非常消耗系統(tǒng)資源,因此嘗試自己通過一次socket連續(xù)傳輸多張圖片

代碼問題記錄(需要代碼的可以直接文末)

在做的過程中發(fā)現(xiàn)了一些問題:

socket在傳一張圖片時(shí)是以二進(jìn)制流的形式傳輸,圖片的二進(jìn)制流比較大,一般一次傳不完,要傳很多次。那么接受者是如何知道什么時(shí)候才停止接收這張圖片呢?那可以讓發(fā)送者在發(fā)圖之前先發(fā)一個(gè)頭信息,告訴接收者這個(gè)二進(jìn)制流有多長(zhǎng),然后接收者通過這個(gè)來判斷是否傳完。

這個(gè)問題是最讓我致命的,由于發(fā)送者先發(fā)了一個(gè)頭信息,使用socket.send()函數(shù),然后發(fā)送圖片也是要用socket.send()函數(shù),接收端使用的是socket.recv(1024)函數(shù),1024是緩存大。麻煩來了,由于發(fā)送者使用連續(xù)的兩個(gè)send,而socket.recv(1024)是有緩存的,他會(huì)把這兩個(gè)信息緩存到一起去,信息頭和圖片信息全被緩存了?。?!這會(huì)直接導(dǎo)致代碼接收邏輯錯(cuò)誤。我的做法是,不能有兩個(gè)send同時(shí)出現(xiàn),那么我就在send中間加一個(gè)recv函數(shù)(阻塞函數(shù)),也就是發(fā)送者每發(fā)一個(gè)消息,接收者就立馬回復(fù)一個(gè)消息,這樣就保證了不會(huì)連續(xù)send

python怎么使用socket高效傳輸視頻數(shù)據(jù)幀 

代碼

由于項(xiàng)目存在兩種數(shù)據(jù)源,一種是可見光,一種是紅外,所以我最開始還要制作一個(gè)信息頭,每次發(fā)送的時(shí)候要告訴接收者這是什么類型的數(shù)據(jù),然后再接收

制作不易,記得給個(gè)贊哈!

客戶端clien.py

服務(wù)器的地址
server_address = ('127.0.0.1', 8000)


def send(dir_name, data_format, file_name):
    # 與接收端建立socket通信
    sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)  # AF_INET(TCP/IP – IPv4)協(xié)議
    sock.connect(server_address)

    # 每次通信都帶一個(gè)通信頭,表明數(shù)據(jù)源的類型(紅外還是可見光),要保存數(shù)據(jù)幀的文件夾名file_name
    # 你可以不要數(shù)據(jù)格式,這里可以定義成你自己的形式,也算是一種安全機(jī)制
    sock.send('{}|{}'.format(data_format, file_name).encode())  # 默認(rèn)編碼 utf-8,發(fā)送文件長(zhǎng)度和文件名
    reply = sock.recv(1024)

    # 按照文件名排序,0.png,1.png
    file_list = os.listdir(dir_name)
    file_list.sort(key=lambda x: int(x[:-4]))
    if 'ok' == reply.decode():  # 確認(rèn)一下服務(wù)器get到文件長(zhǎng)度和文件名數(shù)據(jù)
        i = 0
        print(len(file_list))
        for file_name in file_list:
            data = file_deal(os.path.join(dir_name, file_name))
            sock.send('{}|{}'.format(len(data), file_name).encode())
            sock.recv(1024)
            go = 0
            total = len(data)
            while go < total:  # 發(fā)送文件
                data_to_send = data[go:go + total//2]
                sock.send(data_to_send)
                go += len(data_to_send)
            sock.recv(1024).decode()
            i += 1
            if i < len(file_list):
                sock.send(b'continue')
        sock.send(b'over')
        sock.close()
        sys.exit(0)


def file_deal(file_path):  # 讀取文件的方法
    mes = b''
    try:
        file = open(file_path, 'rb')
        mes = file.read()
    except:
        print('error{}'.format(file_path))
    else:
        file.close()
        return mes

服務(wù)端server.py

LOCAL_IP = '127.0.0.1'  # 本機(jī)測(cè)試使用ip,局域網(wǎng)中使用需更換ip
PORT = 8000  # 隨意指定一個(gè)端口

def server():
    sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)  # socket.AF_INET 指ipv4  socket.SOCK_STREAM 使用tcp協(xié)議
    sock.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)  # 設(shè)置端口
    sock.bind((LOCAL_IP, PORT))  # 綁定端口
    sock.listen(3)  # 監(jiān)聽端口
    while True:
        sc, sc_name = sock.accept()  # 當(dāng)有請(qǐng)求到指定端口是 accept()會(huì)返回一個(gè)新的socket和對(duì)方主機(jī)的(ip,port)
        print('收到{}機(jī)器請(qǐng)求'.format(sc_name))
        info = sc.recv(1024)  # 接受客戶端發(fā)來的協(xié)議頭,區(qū)分?jǐn)?shù)據(jù)源
        # 安全處理:如果不是以這個(gè)協(xié)議頭開始,認(rèn)為是非法接入,就直接斷掉。這里可以自己定義一些安全消息機(jī)制
        print(info)
        try:
            data_format, directory_name = info.decode().split("|")
            sc.send(b'ok')  # 表示收到文件長(zhǎng)度和文件名
        except:
            print('協(xié)議頭不對(duì),自動(dòng)斷開連接')
            sc.close()
            continue

        if not os.path.exists(directory_name):
            os.mkdir(directory_name)
        # 協(xié)議頭正確之后,不斷接收發(fā)來的數(shù)據(jù)幀
        while True:
            head_info = sc.recv(1024)
            # print(data_info)
            length, file_name = head_info.decode().split('|')
            sc.send(b'ok')
            if length and file_name:
                print(file_name)
                newfile = open(os.path.join(directory_name, file_name), 'wb')  # 這里可以使用從客戶端解析出來的文件名
                file = b''
                total = int(length)
                get = 0
                while get < total:  # 接收文件
                    data = sc.recv(total//2)
                    file += data
                    get = get + len(data)
                sc.send(b'ok')
                print('應(yīng)該接收{(diào)},實(shí)際接收{(diào)}'.format(length, len(file)))
                if file:
                    print('actually length:{}'.format(len(file)))
                    newfile.write(file[:])
                    newfile.close()
            reply = sc.recv(1024)
            if reply.decode() == "over":
                break

server()

啟動(dòng)步驟

1.先啟動(dòng)server.py
2.再啟動(dòng)client.py

到此,相信大家對(duì)“python怎么使用socket高效傳輸視頻數(shù)據(jù)幀”有了更深的了解,不妨來實(shí)際操作一番吧!這里是億速云網(wǎng)站,更多相關(guān)內(nèi)容可以進(jìn)入相關(guān)頻道進(jìn)行查詢,關(guān)注我們,繼續(xù)學(xué)習(xí)!

向AI問一下細(xì)節(jié)

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

AI