程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
您现在的位置: 程式師世界 >> 編程語言 >  >> 更多編程語言 >> Python

Python&C++相互混合調用編程全面實戰-30開啟線程調用pyffmpeg擴展庫的Read讀取視幀函數

編輯:Python

作者:虛壞叔叔
博客:https://xuhss.com

早餐店不會開到晚上,想吃的人早就來了!

開啟線程調用pyffmpeg擴展庫的Read讀取視幀函數

一、pyffmpeg擴展庫的Read讀取視幀函數添加

XFFmpeg.h添加讀取視頻幀函數

#pragma once
struct AVFormatContext;
struct AVPacket;
class XFFmpeg
{

public:
// 打開視頻
bool Open(const char *url);
// 讀取一陣視頻 在內部存儲
bool Read();
XFFmpeg();
~XFFmpeg();
int totalms = 0;
protected:
// 解封裝上下文
AVFormatContext *ic = 0;
// 讀取視頻幀
AVPacket *pkt = 0;
};

XFFmpeg.cpp添加函數實現

#include "XFFmpeg.h"
#include <stdio.h>
extern "C" {

#include "libavformat\avformat.h"
}
bool XFFmpeg::Open(const char *url)
{

printf("XFFmpeg::open %s\n", url);
// 打開視頻 解封裝
int re =avformat_open_input(&ic, url, 0, 0);
if (re != 0)
{

char buf[1024] = {
 0 };
av_strerror(re, buf, 1023);
printf("avformat open fail:%s\n", buf);
return false;
}
// 獲取流
avformat_find_stream_info(ic, 0);
// 獲取視頻總時長
this->totalms = ic->duration / (AV_TIME_BASE / 1000);
printf("Total Ms =%d\n", totalms);
return true;
}
bool XFFmpeg::Read()
{

if (!ic)
return false;
// 視頻幀的存儲空間
if (!pkt)
{

// 分配對象空間
pkt = av_packet_alloc();
}
else
{

// 引用計數 -1 清理視頻幀
av_packet_unref(pkt);
}
int re = 0;
bool isFindVideo = false;
// 音頻數據丟掉
for (int i = 0; i < 20; i++)
{

// 讀取一幀數據
re = av_read_frame(ic, pkt);
// 讀取失敗或者讀取到文件結尾
if (re != 0)
{

return false;
}
// 是否是視頻幀
if (pkt->stream_index == 0)
{

isFindVideo = true;
break;
}
// 音頻幀 清理packet
av_packet_unref(pkt);
}
return isFindVideo;
}
XFFmpeg::XFFmpeg()
{

printf("Create XFFmpeg\n");
}
XFFmpeg::~XFFmpeg()
{

printf("Delete XFFmpeg\n");
}

PyFFmpeg.h開放接口,方便python調用

#pragma once
#include<Python.h>
class XFFmpeg;
class PyFFmpeg
{

public:
PyObject_HEAD
XFFmpeg *ff;
//開放給python的函數
public:
static PyObject *Create(PyTypeObject *type, PyObject *args, PyObject *kw);
static int Init(PyFFmpeg*self, PyObject *args, PyObject *kw);
static void Close(PyFFmpeg*self);
static PyObject* Open(PyFFmpeg*self, PyObject*args);
static PyObject *Read(PyFFmpeg*self, PyObject*args);
// 屬性函數 get
static PyObject* GetTotalms(PyFFmpeg*self, void*closure);
};

PyFFmpeg.cpp開放接口的實現添加實現和注冊

#include "PyFFmpeg.h"
#include "XFFmpeg.h"
// 開放給python
PyObject *PyFFmpeg::Create(PyTypeObject *type, PyObject *args, PyObject *kw) {

printf("PyFFmpeg::Create\n");
PyFFmpeg*f = (PyFFmpeg*)type->tp_alloc(type, 0);
f->ff = new XFFmpeg();
return (PyObject *)f;
}
int PyFFmpeg::Init(PyFFmpeg*self, PyObject *args, PyObject *kw)
{

printf("PyFFmpeg::Init\n");
return 0;
}
void PyFFmpeg::Close(PyFFmpeg*self)
{

printf("PyFFmpeg::Close\n");
delete self->ff;
Py_TYPE(self)->tp_free(self);
}
PyObject *PyFFmpeg::Read(PyFFmpeg*self, PyObject*args)
{

if (!self->ff)
Py_RETURN_FALSE;
if (self->ff->Read())
Py_RETURN_TRUE;
Py_RETURN_FALSE;
}
PyObject* PyFFmpeg::Open(PyFFmpeg*self, PyObject*args)
{

const char *url = NULL;
if (!PyArg_ParseTuple(args, "s", &url))
return NULL;
printf("PyFFmpeg::Open %s\n", url);
if (self->ff->Open(url))
Py_RETURN_TRUE;
Py_RETURN_FALSE;
}
PyObject* PyFFmpeg::GetTotalms(PyFFmpeg*self, void*closure)
{

return PyLong_FromLong(self->ff->totalms);
}
// 模塊入口 模塊名稱 pyffmpeg
PyMODINIT_FUNC PyInit_pyffmpeg(void)
{

PyObject *m = NULL;
static PyModuleDef ffmod = {

PyModuleDef_HEAD_INIT,
"pyffmpeg",
"", -1, 0
};
m = PyModule_Create(&ffmod);
// 添加PyFFmpeg_python類
static PyTypeObject type;
memset(&type, 0, sizeof(PyFFmpeg));
type.ob_base = {
 PyObject_HEAD_INIT(NULL) 0 };
type.tp_name = "";
type.tp_basicsize = sizeof(PyFFmpeg);
type.tp_flags = Py_TPFLAGS_DEFAULT | Py_TPFLAGS_BASETYPE;
type.tp_new = PyFFmpeg::Create;
type.tp_init = (initproc)PyFFmpeg::Init;
type.tp_dealloc = (destructor)PyFFmpeg::Close;
static PyMethodDef ffmeth[] = {

{
 "open", (PyCFunction)PyFFmpeg::Open, METH_VARARGS, "" },
{
 "read", (PyCFunction)PyFFmpeg::Read, METH_NOARGS, "" },
{
 NULL }
};
type.tp_methods = ffmeth;
static PyGetSetDef sets[] = {

{
 "totalms", (getter)PyFFmpeg::GetTotalms, 0, 0,0 },
{
 NULL }
};
type.tp_getset = sets;
// 初始化類型
if (PyType_Ready(&type) < 0) {

return NULL;
}
PyModule_AddObject(m, "PyFFmpeg", (PyObject*)&type);
printf("Pyinit_pyffmpeg\n");
return m;
}

pyqt.py添加對接口的調用

isRunning = True
#主函數 在子線程中調用,線程是c++創建
def main():
print("Python main")
global ff;
while isRunning:
print(ff.read());

二、開啟線程調用擴展庫接口

2.1不開啟線程調用

pyplayer.cpp構造函數中調用main函數調用 這裡不開啟線程

PyPlayer::PyPlayer(QWidget *parent)
: QWidget(parent)
{

ui.setupUi(this);
Py_SetPythonHome(L"./");
Py_Initialize();
// 載入模塊
pModule = PyImport_ImportModule("pyqt");
if (!pModule)
{

printf("PyImport import error");
PyErr_Print();
return;
}
// 獲取python配置項 改變窗口的大小和標題
PyObject *conf = PyObject_GetAttrString(pModule, "conf");
if (!conf)
{

cout << "Please set the conf" << endl;
PyErr_Print();
return;
}
PyObject *key = PyUnicode_FromString("width");
int width = PyLong_AsLong(PyDict_GetItem(conf, key));
Py_XDECREF(key);
key = PyUnicode_FromString("height");
int height = PyLong_AsLong(PyDict_GetItem(conf, key));
Py_XDECREF(key);
// 改變窗口標題
key = PyUnicode_FromString("title");
wchar_t title[1024] = {
 0 };
PyUnicode_AsWideChar(PyDict_GetItem(conf, key), title, 1023);
this->setWindowTitle(QString::fromUtf16((char16_t*)title));
Py_XDECREF(key);
if (width > 0 && height > 0)
{

resize(width, height);
}
Py_XDECREF(conf);
// 開放選擇文件的接口給python
static PyMethodDef meths[] = {

{
 "OpenDialog", (PyCFunction)OpenDialog, METH_NOARGS, 0 }
,{
 NULL }
};
int re = PyModule_AddFunctions(pModule, meths);
if (!re)
{

PyErr_Print();
}
// 開啟線程 調用python的 main函數
PyObject*main_fun = PyObject_GetAttrString(pModule, "main");
if (!main_fun || !PyCallable_Check(main_fun))
{

cout << "main_fun get failed!" << endl;
return;
}
if (PyObject_CallObject(main_fun, 0))
{

PyErr_Print();
return;
}
}

運行,你會發現一直輸出false,並且主線程卡死,因為是在一個線程中,pythonmain函數有一個循環,不會退出的循環。

2.2 開啟線程調用讀取視頻

添加一個線程函數run_main,並將執行main函數的代碼放到這個線程中

#include <Python.h>
#include "PyPlayer.h"
#include <iostream>
#include <QFileDialog>
#include <thread>
using namespace std;
static PyObject *pModule = 0;
// 返回選擇的文件路徑
PyObject* OpenDialog(PyObject *self)
{

QString fileName = "";
fileName = QFileDialog::getOpenFileName();
if (fileName.isEmpty())
return PyUnicode_FromString("");
return PyUnicode_FromString(fileName.toStdString().c_str());
}
void PyPlayer::Open() {

cout << "PyPlayer::Open()" << endl;
// 調用Python的open函數
if (!pModule) return;
PyObject *open = PyObject_GetAttrString(pModule, "open");
if (!open || !PyCallable_Check(open))
{

PyErr_Print();
return;
}
PyObject_CallObject(open, 0);
}
void run_main()
{

PyObject*main_fun = PyObject_GetAttrString(pModule, "main");
if (!main_fun || !PyCallable_Check(main_fun))
{

cout << "main_fun get failed!" << endl;
return;
}
if (PyObject_CallObject(main_fun, 0))
{

PyErr_Print();
return;
}
}
PyPlayer::PyPlayer(QWidget *parent)
: QWidget(parent)
{

ui.setupUi(this);
Py_SetPythonHome(L"./");
Py_Initialize();
// 載入模塊
pModule = PyImport_ImportModule("pyqt");
if (!pModule)
{

printf("PyImport import error");
PyErr_Print();
return;
}
// 獲取python配置項 改變窗口的大小和標題
PyObject *conf = PyObject_GetAttrString(pModule, "conf");
if (!conf)
{

cout << "Please set the conf" << endl;
PyErr_Print();
return;
}
PyObject *key = PyUnicode_FromString("width");
int width = PyLong_AsLong(PyDict_GetItem(conf, key));
Py_XDECREF(key);
key = PyUnicode_FromString("height");
int height = PyLong_AsLong(PyDict_GetItem(conf, key));
Py_XDECREF(key);
// 改變窗口標題
key = PyUnicode_FromString("title");
wchar_t title[1024] = {
 0 };
PyUnicode_AsWideChar(PyDict_GetItem(conf, key), title, 1023);
this->setWindowTitle(QString::fromUtf16((char16_t*)title));
Py_XDECREF(key);
if (width > 0 && height > 0)
{

resize(width, height);
}
Py_XDECREF(conf);
// 開放選擇文件的接口給python
static PyMethodDef meths[] = {

{
 "OpenDialog", (PyCFunction)OpenDialog, METH_NOARGS, 0 }
,{
 NULL }
};
int re = PyModule_AddFunctions(pModule, meths);
if (!re)
{

PyErr_Print();
}
// 開啟線程 調用python的 main函數
std::thread t1(run_main);
t1.detach();
}

在執行,界面出來了:

點擊打開按鈕 發現程序崩潰了:

2.3 添加全局GIL鎖,防止軟件崩潰

PyPlayer.cppopen函數中添加gil

void PyPlayer::Open() {

cout << "PyPlayer::Open()" << endl;
// 調用Python的open函數
if (!pModule) return;
PyGILState_STATE gil;
gil = PyGILState_Ensure();
PyObject *open = PyObject_GetAttrString(pModule, "open");
if (!open || !PyCallable_Check(open))
{

PyErr_Print();
PyGILState_Release(gil);
return;
}
PyObject_CallObject(open, 0);
PyGILState_Release(gil);
}

再次運行就不會崩潰了。

2.4 控制台可視化讀取視頻的過程

修改pyqt.py 將讀取視頻的過程通過一個點一個點的輸出出來,這樣就可以驗證讀取的線程是沒有問題的

print("Python PyPlayer")
from pyffmpeg import *
import time
conf = {

"width" : 1280,
"height" : 720,
"title" : "PyPlayer播放器"
}
ff = PyFFmpeg()
def open():
global ff;
print("Python open")
filename = OpenDialog()
if ff.open(filename):
print("open file success!")
else:
print("open file failed!")
print(filename)
isRunning = True
#主函數 在子線程中調用,線程是c++創建
def main():
print("Python main")
global ff;
while isRunning:
#print(ff.read())
re = ff.read()
if re:
print(".", end='', flush = True) #flush輸出緩沖
time.sleep(0.02)
else:
time.sleep(1)

三、總結

  • 本文開啟線程調用pyffmpeg擴展庫的Read讀取視幀函數。
  • 如果覺得文章對你有用處,記得 點贊收藏轉發 一波哦,博主也支持為鐵粉絲制作專屬動態壁紙哦~

往期優質文章分享

  • C++ QT結合FFmpeg實戰開發視頻播放器-01環境的安裝和項目部署
  • 解決QT問題:運行qmake:Project ERROR: Cannot run compiler ‘cl‘. Output:
  • 解決安裝QT後MSVC2015 64bit配置無編譯器和調試器問題
  • Qt中的套件提示no complier set in kit和no debugger,出現黃色感歎號問題解決(MSVC2017)
  • Python+selenium 自動化 - 實現自動導入、上傳外部文件(不彈出windows窗口)

優質教程分享

  • 如果感覺文章看完了不過瘾,可以來我的其他 專欄 看一下哦~
  • 比如以下幾個專欄:Python實戰微信訂餐小程序、Python量化交易實戰、C++ QT實戰類項目 和 算法學習專欄
  • 可以學習更多的關於C++/Python的相關內容哦!直接點擊下面顏色字體就可以跳轉啦!
學習路線指引(點擊解鎖)知識定位人群定位🧡 Python實戰微信訂餐小程序 🧡進階級本課程是python flask+微信小程序的完美結合,從項目搭建到騰訊雲部署上線,打造一個全棧訂餐系統。Python量化交易實戰入門級手把手帶你打造一個易擴展、更安全、效率更高的量化交易系統️ C++ QT結合FFmpeg實戰開發視頻播放器️難度偏高分享學習QT成品的視頻播放器源碼,需要有扎實的C++知識! 游戲愛好者九萬人社區互助/吹水九萬人游戲愛好者社區,聊天互助,白嫖獎品 Python零基礎到入門 Python初學者針對沒有經過系統學習的小伙伴,核心目的就是讓我們能夠快速學習Python的知識以達到入門

資料白嫖,溫馨提示

關注下面卡片即刻獲取更多編程知識,包括各種語言學習資料,上千套PPT模板和各種游戲源碼素材等等資料。更多內容可自行查看哦!


  1. 上一篇文章:
  2. 下一篇文章:
Copyright © 程式師世界 All Rights Reserved