matlabR2008a函數編譯成dll供vc6.0調用的方法

在網上看了篇matlab函數編譯成dll供VC6.0調用的方法,拿來分享一下,裏面有幾處錯誤,糾正一下:

原文地址:http://www.cnblogs.com/vincenzo/archive/2009/09/09/1563467.html

以前做過matlab7與c++的混合編程:將m函數編譯成dll給C++調用,從而加快開發的進度。但是今天在matlab2008b下面又做了一遍,發現matlab又改了很多東西,諸如增加了面向對象的的擴展mwArray,於是做筆記如下。
 
(一) 總體概念
 
matlab提供了豐富的程序接口,除了matlab最初的版本是用fortran寫的之外,後來的版本都是用C寫的,因此matlab很容易和C/C++聯合起來使用。

matlab有着豐富的庫函數,將這些庫函數直接編譯成dll(windows下是dll,linux下是類似於dll的另外一個東西),給C調用,可以加快工程進度。

步驟如下:
1. 設置matlab的編譯器,使用外部的VC或者gcc等編譯器。
2. 編譯m文件成dll
3. 設置VC等的繼承環境,設置C需要用到的靜態庫和動態庫
4. 編寫C調用dll
下面以一個很簡單的加法函數爲例,說明在VC6和matlab2008b下,怎麼做混合編程
 
(二) 設置matlab編譯器(我自己做的時候沒有設置編譯器直接第(三)步,那時再選編譯器) 

過程如下: 1. 在matlab的command window裏面敲入mex –setup,matlab會列出所有電腦上所有的編譯器。 2. 選擇編譯器,這裏選擇2,即VC6 

 

 3.完成之後,敲入mbuild –setup,選擇build工具,matlab會列出所有的build工具,同樣選擇VC6

 (三) 編譯m文件

首先,寫出matlab函數,一個很簡單的加法函數,如下:
function [c] = MyAdd(a, b);
c = a + b;
保存爲MyAdd.m
然後,進行編譯,命令如下:
mcc -W cpplib:libMyAdd -T link:lib MyAdd
其中,mcc是matlab提供的編譯命令(可以理解爲gcc),對於這一點,matlab的幫助說明如下:
-W lib:string link:lib
其中-W是控制編譯之後的封裝格式,cpplib,是指編譯成C++的lib,cpplib冒號後面是指編譯的庫的名字,-T表示目標,link:lib表示要連接到一個庫文件的目標,目標的名字是後面的MyAdd,即你寫的m函數的名字。
運行完之後,你的目錄下面,將會出現以下一些文件:
libmyadd.cpp
libmyadd.exp
libmyadd.dll
libmyadd.exports
libmyadd.h
libmyadd.prj
libmyadd_mcc_component_data.c
mccExcludedFiles.log
readme.txt
還有一個文件libmyadd.lib
具體每一個文件的意思,請查幫助,這裏只說其中很有用的幾個:
libMyAdd.dll是編譯好的動態庫,MyAdd這個函數封裝在裏面,libMyAdd.h是對應的頭文件,在C裏面調用這個函數的時候,需要include這個頭文件,在這個頭文件的最後,你可以看到下面的內容:
extern LIB_libmyadd_CPP_API void MW_CALL_CONV myadd(int nargout, mwArray& c
, const mwArray& a
, const mwArray& b);
這就是MyAdd的C函數的聲明。
nargout表示輸出變量的個數
a,b,c就是對應了m函數的幾個變量
注意,a,b,c三個變量的類型是mwArray,這是matlab提供的一個特殊的變量類型(在7.0的版本,編譯之後,變量類型是mxArray)。mwArray是一個類,具體可以查幫助,當然後面也會講到。
 
(四) 設置VC環境
 
有了上面的準備,需要設置VC環境,好讓lib和dll能被成功的調用。
1. 首先建立一個控制檯工程(這個我就不詳細講了,太easy了)
2. 然後,在tool-option-directory裏面設置用到的lib和include頭文件的位置
首先是include頭文件的位置,看最後一行(這個根據每一個人的安裝目錄可能不一樣,自己去找吧)

 

 然後是lib的位置,不盡包括matlab的靜態庫的位置,還包括剛剛編譯出來的lib的位置,見圖的後面兩個,不用我解釋吧

 

你可以打開這兩個目錄,看看裏面到底有哪些頭文件和lib,首先來講,我們這裏要用到的頭文件包括:

#include "mclmcr.h"
#include "matrix.h"
#include "mclcppclass.h"
#include "libmyadd.h"
lib包括
mclmcrrt.lib libmx.lib libmat.lib mclmcr.lib這裏少了個剛剛編譯的libmyadd.lib
你可以到具體的路徑下看一看,有沒有上面幾個lib和頭文件
路徑設好了,需要制定具體用到的lib了,在project Setting-link-Object/library modules裏面加入上面提到的幾個lib(具體幾個lib到底做什麼用的,這裏不講了,看幫助,或者你試驗刪去幾個lib能不能工作,就大概可以猜得到他們是幹什麼的了) 

 
3. 將編譯好的dll複製到VC工程的Debug或者Release目錄下,以使得dll可以被找到。還要把編譯生成的libmyadd.h文件拷貝到VC工程裏
 
(五) 寫Cpp調用dll
 
所有任務完成之後,開始寫Cpp試驗調用編譯之後的函數了。
首先,包含應有的頭文件:
 
  1. #include "mclmcr.h"   
  2. #include "matrix.h"   
  3. #include "mclcppclass.h"   
  4. #include "libmyadd.h"   
  5. #include <stdio.h>   
  6. #include <iostream.h>   
  7.  
  8. //然後,寫main函數:   
  9. int main(void)   
  10. {   
  11.     double a = 6;   
  12.     double b = 9;   
  13.     double c;   
  14.       
  15.     // initialize lib,這裏必須做初始化!   
  16.     if( !libmyaddInitialize())   
  17.     {   
  18.         std::cout << "Could not initialize libMyAdd!" << std::endl;   
  19.         return -1;   
  20.     }   
  21.       
  22.     // 爲變量分配內存空間,可以查幫助mwArray   
  23.     mwArray mwA(1, 1, mxDOUBLE_CLASS); // 1,1表示矩陣的大小(所有maltab只有一種變量,就是矩陣,爲了和Cpp變量接軌,設置成1*1的矩陣,mxDOUBLE_CLASS表示變量的精度)   
  24.     mwArray mwB(1, 1, mxDOUBLE_CLASS);   
  25.     mwArray mwC(1, 1, mxDOUBLE_CLASS);   
  26.       
  27.     // set data,不用我解釋了吧,很簡單的,調用類裏面的SetData函數給類賦值   
  28.     mwA.SetData(&a, 1);   
  29.     mwB.SetData(&b, 1);   
  30.       
  31.     // using my add,掉我自己寫的函數   
  32.     myadd(1, mwC, mwA, mwB);   
  33.       
  34.     // get data,不用我解釋了吧,很簡單的,調用類裏面的Get函數獲取取函數返回值   
  35.     c = mwC.Get(1, 1);   
  36.     printf("c is %f\n", c);   
  37.       
  38.     // 後面是一些終止調用的程序   
  39.     // terminate the lib   
  40.     libmyaddTerminate();   
  41.       
  42.     // terminate MCR   
  43.     mclTerminateApplication();   
  44.       
  45.     return 0;   
  46.       
  47. }   
 
運行結果如下:

 

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章