日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程语言 > c/c++ >内容正文

c/c++

C++和MATLAB混合编程-DLL篇

發布時間:2025/3/21 c/c++ 37 豆豆
生活随笔 收集整理的這篇文章主要介紹了 C++和MATLAB混合编程-DLL篇 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

先小話一下DLL,DLL是動態鏈接庫,是源代碼編譯后的二進制庫文件和程序接口,和靜態鏈接庫不同的是,程序在編譯時并不鏈接動態鏈接庫的執行體,而是在文件中保留一個調用標記,在程序運行時才將動態鏈接庫文件加載入內存。并且DLL在運行時是共享的,即當多個程序調用時,內存中也只保持一份動態鏈接庫。

動態鏈接庫的調用有顯式和隱式兩種方式。

隱式鏈接需要用到我們前面生成的plotdata.c,plotdata.h,plotdata.lib以及plotdata.dll文件。
首先將plotdata.c,plotdata.h加入工程中,注意在需要用到函數的文件加入#include “plotdata.h”。
之后鏈接輸入項中寫上plotdata.lib。右擊工程->Propertites->Link->Input->Additional Dependecies中加上plotdata.lib(也就是在調用MATLAB引擎時填寫libmat.lib、libeng.lib等的地方)注意plotdata.lib也需要放在你的工程下,或者寫全路徑,如"D:\data\plotadata.lib",需要加引號。
這樣在你的代碼中就可以直接用plotdata.h中的接口函數了。

另顯式鏈接的方式:所謂“顯式”說白了就是在代碼中寫出來我要調用這個DLL。

首先我們需要定義一個函數類型,方便我們后面進行函數的強制類型轉換。我們可以在plotdata.h中找到我們將要使用的函數plotdata,他的函數聲明如下:

[cpp]?view plaincopy
  • extern?LIB_plotdata_CPP_API?void?MW_CALL_CONV?plotdata(const?mwArray&?rgbData);??

  • 忽略那些復雜的宏定義,模仿著定義我們自己的函數類型:

    [cpp]?view plaincopy
  • typedef?void?(*HMAT)(const?mwArray&?rgbData);??

  • 之后在代碼中顯式鏈接plotdata.dll

    [cpp]?view plaincopy
  • HINSTANCE?hDLL=NULL;//DLL句柄??
  • hDLL=LoadLibrary("plotdata.dll");??
  • HMAT?plotData=(HMAT)GetProcAddress(hDLL,"plotdata");??
  • ????????????????//第一個參數為DLL句柄,第二個為要加載的函數名??
  • 之后便可直接在代碼中直接使用函數plotData了。這種顯式鏈接只需要plotdata.dll文件即可~
    下面我們來看一下生成的函數接口
    [cpp]?view plaincopy
  • extern?LIB_plotdata_C_API???
  • bool?MW_CALL_CONV?plotdataInitializeWithHandlers(??
  • ???????mclOutputHandlerFcn?error_handler,???
  • ???????mclOutputHandlerFcn?print_handler);??
  • ??
  • extern?LIB_plotdata_C_API???
  • bool?MW_CALL_CONV?plotdataInitialize(void);??
  • ??
  • extern?LIB_plotdata_C_API???
  • void?MW_CALL_CONV?plotdataTerminate(void);??
  • ??
  • extern?LIB_plotdata_C_API???
  • void?MW_CALL_CONV?plotdataPrintStackTrace(void);??
  • ??
  • extern?LIB_plotdata_C_API???
  • bool?MW_CALL_CONV?mlxPlotdata(int?nlhs,?mxArray?*plhs[],?int?nrhs,?mxArray?*prhs[]);??
  • ??
  • extern?LIB_plotdata_C_API???
  • long?MW_CALL_CONV?plotdataGetMcrID();??
  • ??
  • extern?LIB_plotdata_C_API?bool?MW_CALL_CONV?mlfPlotdata(mxArray*?rgbData);??
  • 這是plotdata.h中主要的函數。plotdataInitialize(void)可以看出是初始化的函數。如果是隱式鏈接DLL最好先調用此函數,判斷返回值否則很可能加載不到dll,而在顯式鏈接時,如果沒有加載函數成功,也不會直接報錯,但我們可以在單步調試時看函數是否為分配了內存(即是否為null)。plotdataTerminate(void)是終止動態鏈接庫的函數。
    mlxPlotdatat與mlfPlotdata是最關鍵的兩個接口,也是我們要加載的函數。他們執行的功能與m文件中plotdata函數是一樣的。兩個函數輸入參數不同:
    mlxPlotdata(int nlhs, mxArray *plhs[], int nrhs, mxArray *prhs[]); 其中 nlhs,plhs分別表示輸出參數的個數及輸出參數的mxArray數組;nrhs,prhs表示輸入參數的個數及輸入參數數組。(這個函數有點通用的感覺……)
    mlfPlotdata(mxArray* rgbData); 就簡單的多,基本和m文件中你定義的plotdata函數是一樣的(我的plotdata定義為 function []=plotdata(rgbData))
    所以一般在程序中加載的是mlf開頭的函數。
    這里需要提的是我編譯生成的是C的動態鏈接庫。如果是生成C++的動態鏈接庫,生成的接口函數也帶有一個mlx開頭的函數,即
    bool MW_CALL_CONV mlxPlotdata(int nlhs, mxArray *plhs[], int nrhs, mxArray *prhs[])
    但是另一個函數是不帶有mlf的,直接為
    void MW_CALL_CONV plotdata(const mwArray& rgbData)
    而且輸入參數不是mxArray數組,而是mwArray數組,這也是C和C++與MATLAB混合編程時最主要的不同(下篇再詳細說)
    但是我在嘗試C++動態鏈接時一直沒有成功。后來看到生成的cpp文件同c文件一樣也有一個 extend "C"{},這是C++為了與C兼容而提供的一個關鍵字,C++編譯器將會在extend "C"的大括號內部代碼當做C語言代碼處理,這讓我很困惑……而如果注釋掉又會報連接錯誤
    想來可能是MATLAB對C++編譯支持并不好(他自帶的lcc編譯器是只能編譯成C的接口)總之沒有嘗試成功,暫時在程序中都用C的動態鏈接了。


    (轉載請注明作者和出處:http://blog.csdn.net/xiaowei_cqu?未經允許請勿用于商業用途)


    [cpp]?view plaincopy
  • <pre?code_snippet_id="1

  • 總結

    以上是生活随笔為你收集整理的C++和MATLAB混合编程-DLL篇的全部內容,希望文章能夠幫你解決所遇到的問題。

    如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。