詳解如何實現(xiàn)C#和Python間實時視頻數(shù)據(jù)交互
我們在做RTSP|RTMP播放的時候,遇到好多開發(fā)者,他們的視覺算法大多運行在python下,需要高效率的實現(xiàn)C#和Python的視頻數(shù)據(jù)交互,常用的方法如下:
方法一:通過HTTP請求傳輸視頻數(shù)據(jù)
服務器端(Python)
使用Flask或Django等Web框架創(chuàng)建一個HTTP服務器,通過API接口提供視頻數(shù)據(jù)。
from flask import Flask, send_file, request, jsonify
import cv2
import io
import base64
app = Flask(__name__)
def get_video_frame():
# 讀取視頻幀
cap = cv2.VideoCapture('your_video.mp4')
ret, frame = cap.read()
cap.release()
if not ret:
return None, "No more frames"
# 轉(zhuǎn)換為RGB格式
frame_rgb = cv2.cvtColor(frame, cv2.COLOR_BGR2RGB)
# 轉(zhuǎn)換為字節(jié)流
ret, buffer = cv2.imencode('.jpg', frame_rgb)
if not ret:
return None, "Could not encode frame"
# 將字節(jié)流轉(zhuǎn)換為base64字符串
img_str = base64.b64encode(buffer).decode('utf-8')
return img_str, 200
@app.route('/video_frame', methods=['GET'])
def video_frame():
img_str, status_code = get_video_frame()
if img_str is None:
return jsonify({"error": status_code}), status_code
return jsonify({"image": img_str})
if __name__ == '__main__':
app.run(debug=True)客戶端(C#)
使用HttpClient從Python服務器獲取視頻幀,并將其顯示在PictureBox中。
using System;
using System.Drawing;
using System.Drawing.Imaging;
using System.IO;
using System.Net.Http;
using System.Windows.Forms;
public class VideoForm : Form
{
private PictureBox pictureBox;
private HttpClient httpClient;
public VideoForm()
{
pictureBox = new PictureBox
{
Dock = DockStyle.Fill,
SizeMode = PictureBoxSizeMode.StretchImage
};
this.Controls.Add(pictureBox);
httpClient = new HttpClient();
Timer timer = new Timer();
timer.Interval = 100; // Adjust the interval as needed
timer.Tick += Timer_Tick;
timer.Start();
}
private async void Timer_Tick(object sender, EventArgs e)
{
try
{
HttpResponseMessage response = await httpClient.GetAsync("http://localhost:5000/video_frame");
response.EnsureSuccessStatusCode();
string jsonResponse = await response.Content.ReadAsStringAsync();
dynamic jsonData = Newtonsoft.Json.JsonConvert.DeserializeObject(jsonResponse);
string base64String = jsonData.image;
byte[] imageBytes = Convert.FromBase64String(base64String);
using (MemoryStream ms = new MemoryStream(imageBytes))
{
Image image = Image.FromStream(ms);
pictureBox.Image = image;
}
}
catch (Exception ex)
{
MessageBox.Show($"Error: {ex.Message}");
}
}
[STAThread]
public static void Main()
{
Application.EnableVisualStyles();
Application.Run(new VideoForm());
}
}方法二:通過ZeroMQ傳輸視頻數(shù)據(jù)
ZeroMQ是一種高性能的異步消息庫,適用于需要低延遲和高吞吐量的應用。
服務器端(Python)
使用pyzmq庫發(fā)送視頻幀。
import zmq
import cv2
import numpy as np
context = zmq.Context()
socket = context.socket(zmq.PUB)
socket.bind("tcp://*:5555")
cap = cv2.VideoCapture('your_video.mp4')
while True:
ret, frame = cap.read()
if not ret:
break
frame_rgb = cv2.cvtColor(frame, cv2.COLOR_BGR2RGB)
frame_bytes = frame_rgb.tobytes()
socket.send_string("FRAME", zmq.SNDMORE)
socket.send(frame_bytes, flags=0, copy=False)客戶端(C#)
使用NetMQ庫接收視頻幀。
using System;
using System.Drawing;
using System.Drawing.Imaging;
using System.Runtime.InteropServices;
using System.Threading.Tasks;
using System.Windows.Forms;
using NetMQ;
using NetMQ.Sockets;
public class VideoForm : Form
{
private PictureBox pictureBox;
private SubscriberSocket subscriber;
public VideoForm()
{
pictureBox = new PictureBox
{
Dock = DockStyle.Fill,
SizeMode = PictureBoxSizeMode.StretchImage
};
this.Controls.Add(pictureBox);
var address = "tcp://localhost:5555";
using (var context = NetMQContext.Create())
{
subscriber = context.CreateSubscriberSocket();
subscriber.Connect(address);
subscriber.Subscribe("FRAME");
Task.Run(() => ReceiveFramesAsync(subscriber));
}
}
private async Task ReceiveFramesAsync(SubscriberSocket subscriber)
{
while (true)
{
var topic = await subscriber.ReceiveFrameStringAsync();
var frameBytes = await subscriber.ReceiveFrameBytesAsync();
if (topic == "FRAME")
{
int width = 640; // Adjust based on your video resolution
int height = 480; // Adjust based on your video resolution
int stride = width * 3;
Bitmap bitmap = new Bitmap(width, height, PixelFormat.Format24bppRgb);
BitmapData bitmapData = bitmap.LockBits(new Rectangle(0, 0, width, height), ImageLockMode.WriteOnly, bitmap.PixelFormat);
Marshal.Copy(frameBytes, 0, bitmapData.Scan0, frameBytes.Length);
bitmap.UnlockBits(bitmapData);
pictureBox.Image = bitmap;
}
}
}
[STAThread]
public static void Main()
{
Application.EnableVisualStyles();
Application.Run(new VideoForm());
}
}方法三:通過共享內(nèi)存或文件
如果C#和Python運行在同一臺機器上,可以通過共享內(nèi)存或文件系統(tǒng)進行數(shù)據(jù)交換。這種方法相對簡單,但性能可能不如前兩種方法。
以上就是詳解如何實現(xiàn)C#和Python間實時視頻數(shù)據(jù)交互的詳細內(nèi)容,更多關(guān)于C#和Python視頻數(shù)據(jù)交互的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
C#難點逐個擊破(8):可空類型System.Nullable
null值用來表示數(shù)據(jù)類型未被賦予任何值,它是一種引用類型;void表示沒有類型,或者說是沒有任何值。null與void的區(qū)別可以認為void是根本沒有,而null是一個空箱子,里面什么都沒有。2010-02-02
C#中Foreach循環(huán)遍歷的本質(zhì)與枚舉器詳解
這篇文章主要給大家介紹了關(guān)于C#中Foreach循環(huán)遍歷本質(zhì)與枚舉器的相關(guān)資料,foreach循環(huán)用于列舉出集合中所有的元素,foreach語句中的表達式由關(guān)鍵字in隔開的兩個項組成,本文通過示例代碼介紹的非常詳細,需要的朋友可以參考下2021-08-08
C#使?XmlReader和XmlWriter操作XML?件
這篇文章介紹了C#使?XmlReader和XmlWriter操作XML?件的方法,文中通過示例代碼介紹的非常詳細。對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下2022-06-06
C#使用log4net結(jié)合sqlite數(shù)據(jù)庫實現(xiàn)記錄日志
因為結(jié)構(gòu)化的數(shù)據(jù)庫存儲的日志信息,可以寫專門的軟件讀取歷史日志信息,通過各種條件篩選,可操作性極大增強,有這方面需求的開發(fā)人員可以考慮,本文給大家介紹了C#使用log4net結(jié)合sqlite數(shù)據(jù)庫記錄日志,需要的朋友可以參考下2024-10-10

