【摘要】 在windows下做應(yīng)用開(kāi)發(fā)時(shí),經(jīng)常需要多種不同的語(yǔ)言混合編程。比如:利用Qt開(kāi)發(fā)一個(gè)動(dòng)態(tài)庫(kù),給C#調(diào)用。 當(dāng)前的需求是: 利用Qt開(kāi)發(fā)一個(gè)工具庫(kù),給C#調(diào)用,來(lái)完成一些特殊處理。 需要Qt生成一個(gè)動(dòng)態(tài)庫(kù)(dll),給C#加載調(diào)用,并且還需要設(shè)置回調(diào),方便C#知道Qt運(yùn)行時(shí),輸出內(nèi)部的一些實(shí)時(shí)消息。 這個(gè)Qt庫(kù)是不需要界面的,只是一個(gè)單純的庫(kù),提供方法給C#調(diào)用,完成指定的功能即可。
1. 前言
在windows下做應(yīng)用開(kāi)發(fā)時(shí),經(jīng)常需要多種不同的語(yǔ)言混合編程。比如:利用Qt開(kāi)發(fā)一個(gè)動(dòng)態(tài)庫(kù),給C#調(diào)用。
當(dāng)前的需求是: 利用Qt開(kāi)發(fā)一個(gè)工具庫(kù),給C#調(diào)用,來(lái)完成一些特殊處理。
需要Qt生成一個(gè)動(dòng)態(tài)庫(kù)(dll),給C#加載調(diào)用,并且還需要設(shè)置回調(diào),方便C#知道Qt運(yùn)行時(shí),輸出內(nèi)部的一些實(shí)時(shí)消息。 這個(gè)Qt庫(kù)是不需要界面的,只是一個(gè)單純的庫(kù),提供方法給C#調(diào)用,完成指定的功能即可。
比如:視頻加水印,圖片模糊處理,圖片鏡像,視頻特效等等。
接下來(lái)就利用一個(gè)小Demo來(lái)演示一下整個(gè)流程。
當(dāng)前我的開(kāi)發(fā)環(huán)境:
VS版本: VS2017
Qt版本: Qt5.12.6
在此之前,需要先給vs2017搭建QT的環(huán)境,也就是安裝Qt插件。這個(gè)流程在之前的文章里已經(jīng)有詳細(xì)介紹,可以翻閱。
2. 創(chuàng)建Qt項(xiàng)目
2.1 新建工程
到此,工程模板創(chuàng)建成功。
2.2 編寫函數(shù)接口
為了外部能夠調(diào)用,需要提供函數(shù)接口給外部調(diào)用,我這里采用編寫個(gè)簡(jiǎn)單的Demo來(lái)進(jìn)行演示。
我這里寫了1個(gè)接口,這個(gè)接口用于圖片的縮放,形參里最后一個(gè)參數(shù)是設(shè)置回調(diào)函數(shù)指針,用于回調(diào)給C#輸出一些提示,一些其他數(shù)據(jù)。
//回調(diào)函數(shù)指針
typedef void(*CallBackFunction_p)(const char *p);
//圖片縮放接口
extern "C" _declspec(dllimport) int ImageZoom(int w,int h,char* image_path,CallBackFunction_p func_p);
.h文件新增的代碼如下:
因?yàn)橐幚韴D片,這里加入Qt需要使用的頭文件。
.c文件新增代碼如下:
QString __NewFile;
//圖片縮放接口
int ImageZoom(int w, int h, char* image_path, CallBackFunction_p func_p)
{
QImage img(image_path);
QImage result = img.scaled(w,h, Qt::IgnoreAspectRatio, Qt::SmoothTransformation);
__NewFile = QString("%1_zoom.png").arg(QFileInfo(image_path).baseName());
int state=result.save(__NewFile);
//調(diào)用回調(diào),通知C#新圖片生成的路徑
func_p(__NewFile.toStdString().c_str());
return state;
}
2.3 編譯生成動(dòng)態(tài)庫(kù)
編譯成功后生成的庫(kù)文件如下:
2.4 打包依賴文件
生成庫(kù)之后,不能直接拿去調(diào)用,還需要找到這個(gè)庫(kù)所需要的其他庫(kù)文件,放到一起再拷貝到C#目錄下,才可以正常調(diào)用運(yùn)行。
因?yàn)槲矣玫氖?2位編譯器編譯的庫(kù),點(diǎn)擊windows狀態(tài)欄左下角的window圖標(biāo),彈出選項(xiàng)欄,找到對(duì)應(yīng)的控制臺(tái),點(diǎn)擊進(jìn)入。
C:\Qt\Qt5.12.6\5.12.6\msvc2017>cd /d D:\out\VS2017_Test\QtClassLibrary1\Release
D:\out\VS2017_Test\QtClassLibrary1\Release>windeployqt.exe QtClassLibrary1.dll
利用Qt的windeployqt.exe 工具,自動(dòng)搜索拷貝依賴庫(kù)。
依賴庫(kù)搜索完成。
3. 創(chuàng)建C#項(xiàng)目
3.1 新建工程
創(chuàng)建好的工程模板如下:
3.2 編寫代碼調(diào)用Qt接口
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Runtime.InteropServices;
namespace ConsoleApp1
{
class Program
{
[DllImport("QtClassLibrary1.dll", EntryPoint = "ImageZoom", CallingConvention = CallingConvention.Cdecl, CharSet = CharSet.Auto)]
public extern static int ImageZoom(int w,int h,IntPtr Path, CallbackDelegate callback);
//定義委托
[UnmanagedFunctionPointer(CallingConvention.Cdecl)]
public delegate void CallbackDelegate(IntPtr Path);
//接收C++回調(diào)數(shù)據(jù)
static void CallBackFunction(IntPtr Path)
{
Console.WriteLine("C++傳出來(lái)的回調(diào):" + Marshal.PtrToStringAnsi(Path));
}
static void Main(string[] args)
{
string text = "C:\\Users\\11266\\Pictures\\20220425103841.png";
int r_code = ImageZoom(100,100,Marshal.StringToHGlobalAnsi(text), CallBackFunction);
Console.WriteLine("執(zhí)行狀態(tài):" + r_code);
Console.ReadKey();
}
}
}
寫完代碼,直接運(yùn)行,會(huì)報(bào)錯(cuò)找不到模塊。很正常,因?yàn)榇a里填寫的庫(kù)是當(dāng)前程序運(yùn)行路徑,現(xiàn)在路徑下并沒(méi)有庫(kù)文件,接下來(lái)需要拷貝庫(kù)到運(yùn)行目錄下即可。
3.3 拷貝庫(kù)文件
3.4 再次運(yùn)行
圖片已經(jīng)縮放成功:
到此,C#調(diào)用Qt生成的庫(kù)調(diào)用完成。
4. 信號(hào)槽的問(wèn)題
如果在庫(kù)里面需要使用到Qt信號(hào)與槽函數(shù),需要手動(dòng)啟用事件循環(huán)。
定義一個(gè)全局變量,初始化QCoreApplication
static int argc = 1;
static char arg0[] = "";
static char * argv[] = { arg0, nullptr };
QCoreApplication app(argc, argv);
然后在需要啟動(dòng)事件循環(huán)的地方,執(zhí)行:
//開(kāi)始事件轉(zhuǎn)換
app.exec();
在合理的地方進(jìn)行退出,事件循環(huán): (比如:槽函數(shù)響應(yīng)里)
app.quit();
貼出個(gè)定時(shí)器例子:
#include
#include
static int argc = 1;
static char arg0[] = "";
static char * argv[] = { arg0, nullptr };
QCoreApplication app(argc, argv);
void MediaToolLibrary::update_time()
{
time_cnt++;
if (time_cnt >= 5)app.quit(); //退出事件循環(huán)
qDebug() << "時(shí)間進(jìn)行中:" << time_cnt++;
}
//要素塊導(dǎo)出
void MediaToolLibrary::VideoElementExport(QString MediaInfo, CallBackFunction_p func_p)
{
time_cnt = 0;
timer = new QTimer(this);
connect(timer, SIGNAL(timeout()), this, SLOT(update_time()));
timer->start(1000);
//開(kāi)始事件轉(zhuǎn)換
app.exec();
qDebug() << "時(shí)間到達(dá).....";
}
-
嵌入式
+關(guān)注
關(guān)注
5089文章
19168瀏覽量
306738 -
WINDOWS
+關(guān)注
關(guān)注
4文章
3556瀏覽量
89068 -
Qt
+關(guān)注
關(guān)注
1文章
308瀏覽量
37999
發(fā)布評(píng)論請(qǐng)先 登錄
相關(guān)推薦
評(píng)論