python_socket 圖像網絡傳輸


本代碼主要實現的是利用網絡傳輸圖片,由於在遠程進行rknn推理時,讀取的圖像數據傳輸時間很長,所以編碼成JPG格式的圖片傳輸,從而減少傳輸時間。

所以本質還是希望讀者借此代碼可以了解python的socket編程。代碼意義已在注釋中詳細說明,僅供參考。使用時請注意修改IP地址和端口號。

以下是服務端代碼

import socket
import time
import cv2
import numpy
 
def ReceiveVideo():
    #IP地址'0.0.0.0'為等待客戶端連接
    address = ('0.0.0.0', 8002)
    #建立socket對象,參數意義見https://blog.csdn.net/rebelqsp/article/details/22109925
    #socket.AF_INET:服務器之間網絡通信 
    #socket.SOCK_STREAM:流式socket , for TCP
    s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
    #將套接字綁定到地址, 在AF_INET下,以元組(host,port)的形式表示地址.
    s.bind(address)
    #開始監聽TCP傳入連接。參數指定在拒絕連接之前,操作系統可以掛起的最大連接數量。該值至少為1,大部分應用程序設為5就可以了。
    s.listen(1)

    def recvall(sock, count):
        buf = b''#buf是一個byte類型
        while count:
            #接受TCP套接字的數據。數據以字符串形式返回,count指定要接收的最大數據量.
            newbuf = sock.recv(count)
            if not newbuf: return None
            buf += newbuf
            count -= len(newbuf)
        return buf
    
    #接受TCP連接並返回(conn,address),其中conn是新的套接字對象,可以用來接收和發送數據。addr是連接客戶端的地址。
    #沒有連接則等待有連接
    conn, addr = s.accept()
    print('connect from:'+str(addr))
    
    while 1:
        start = time.time()#用於計算幀率信息
        length = recvall(conn,16)#獲得圖片文件的長度,16代表獲取長度
        stringData = recvall(conn, int(length))#根據獲得的文件長度,獲取圖片文件
        data = numpy.frombuffer(stringData, numpy.uint8)#將獲取到的字符流數據轉換成1維數組
        decimg=cv2.imdecode(data,cv2.IMREAD_COLOR)#將數組解碼成圖像
        cv2.imwrite("./test.jpg",decimg)
        #print(decimg)
        #cv2.imshow('SERVER',decimg)#顯示圖像

        end = time.time()
        seconds = end - start
        fps  = 1/seconds;
        conn.send(bytes(str(int(fps)),encoding='utf-8'))
        #k = cv2.waitKey(10)&0xff
        #if k == 27:
        #    break
    s.close()
    #cv2.destroyAllWindows()
     
if __name__ == '__main__':
    ReceiveVideo()

以下是客戶端代碼

import socket
import cv2
import numpy
import time
import sys 
 
def SendVideo():
    #建立sock連接
    #address要連接的服務器IP地址和端口號
    address = ('192.168.6.72', 8002)
    try:
        #建立socket對象,參數意義見https://blog.csdn.net/rebelqsp/article/details/22109925
        #socket.AF_INET:服務器之間網絡通信 
        #socket.SOCK_STREAM:流式socket , for TCP
        sock = socket.socket(socket.AF_INET,socket.SOCK_STREAM)
        #開啟連接
        sock.connect(address)
    except socket.error as msg:
        print(msg)
        sys.exit(1)
 
    #建立圖像讀取對象
    capture = cv2.VideoCapture(0)
    #讀取一幀圖像,讀取成功:ret=1 frame=讀取到的一幀圖像;讀取失敗:ret=0
    ret, frame = capture.read()
    #壓縮參數,后面cv2.imencode將會用到,對於jpeg來說,15代表圖像質量,越高代表圖像質量越好為 0-100,默認95
    encode_param=[int(cv2.IMWRITE_JPEG_QUALITY),15]
 
    while ret:
        #停止0.1S 防止發送過快服務的處理不過來,如果服務端的處理很多,那么應該加大這個值
        time.sleep(0.01)
        #cv2.imencode將圖片格式轉換(編碼)成流數據,賦值到內存緩存中;主要用於圖像數據格式的壓縮,方便網絡傳輸
        #'.jpg'表示將圖片按照jpg格式編碼。
        result, imgencode = cv2.imencode('.jpg', frame, encode_param)
        #建立矩陣
        data = numpy.array(imgencode)
        #將numpy矩陣轉換成字符形式,以便在網絡中傳輸
        stringData = data.tostring()
        
        #先發送要發送的數據的長度
        #ljust() 方法返回一個原字符串左對齊,並使用空格填充至指定長度的新字符串
        sock.send(str.encode(str(len(stringData)).ljust(16)));
        #發送數據
        sock.send(stringData);
        #讀取服務器返回值
        receive = sock.recv(1024)
        if len(receive):print(str(receive,encoding='utf-8'))
        #讀取下一幀圖片
        ret, frame = capture.read()
        if cv2.waitKey(10) == 27:
            break
    sock.close()
    
if __name__ == '__main__':
    SendVideo()

參考:https://blog.csdn.net/Barry_J/article/details/98727933


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM