.m文件編譯成com/.Net Assembly以及C#反射

運行環境:Windows 8 Release Preview(Build 8400),Matlab R2012a,Visual Studio 2010

把matlab的.m文件編譯生成com組件或是.net assembly的方法:

Matlab的界面輸入mbuild -setup,選擇VC++編譯器,然後輸入deploytool,新建deploytool project,

如果選擇Generic COM Component,則生成com組件,在C#中只能靜態加載,即在C#中添加引用,然後using XXX,然後在程序中使用;

如果選擇.Net Assembly,則生成C#能動態加載的.Net組件,可以用C#的反射機制動態加載。

當然可以在命令行中使用mcc命令實現,mcc的詳細命令參見http://www.cnitblog.com/edaiqingwa/archive/2006/12/12/20413.aspx

在matlab中用mcc命令把.m文件編譯成.Net Assembly組件的命令如下:mcc -W ‘dotnet:mcccom,mcccomclass,0.0,private’ -T link:lib -d ‘des_content’ xxx.m

mcccom爲生成的程序集名,也就是C#中建立的namespace,mcccomclass爲生成的類名.-T link: lib 選項指定作爲一C共享庫輸出 -d 輸出路徑, xxx.m爲matlab文件,

其中類名和.m文件中的函數名別一樣。

在dos下用mcc命令把.m文件編譯成.Net Assembly組件的命令如下:mcc -W "dotnet:mcccom,mcccomclass,0.0,private" -T link:lib -d "des_content" xxx.m;

在C#中的命令如下:System.Diagnostics.Process.Start("cmd.exe", "/c mcc -W \"dotnet:mcccom,mcccomclass,0.0,private\" -T link:lib -d \"D:\\temp\\dll\" " + m_name);

其中保留cmd 窗口命令爲System.Diagnostics.Process.Start("cmd.exe", "/k dir /a"),不保留cmd 窗口命令爲System.Diagnostics.Process.Start("cmd.exe", "/c dir /a");

其中在C#中需要注意的是引號問題,筆者就因爲引號處理的不好而一直編譯不成功,跳出“unable to locate XXX to matlab path”的錯誤,還一直修改matlab path,甚至還修改環境變量,結果卻是參數引號的問題,浪費了好多時間,所以務必保證cmd.exe中的參數正確。

m文件編譯成.Net Assembly後生成的dll就可以用C#的反射動態加載了,首先簡單地說一下反射的概念。

最近在做關於動態加載dll文件時接觸到了反射的概念,一開始覺得好高深,熟悉了一會兒覺得也就那樣,當然反射的功能的確是好高深的。

反射是一種機制,通過這種機制我們可以知道一個未知類型的類型信息.比如,有一個dll文件,但我們不知道里面是什麼東西,但我們知道它可以實現什麼功能,也就是大致知道里面有個實現

此功能的方法,我們便可以用反射來獲得dll中的任何類和方法,然後進行調用。

先寫一個類:

namespace ToLower
 {
     public class Tolower
     {
         public string exe()
         {
              return "tolower";
         }
         
         public string execute(string s)
         {
             return s.ToLower();
         }
     }
     public class tt
     { }
 }

然後通過反射獲得類以及方法名,進而調用其方法。代碼如下:

string news;
 //string[] paths = Directory.GetFiles(@"C:\Users\chy\Desktop\textdll\", "*.dll");
 //foreach (string path in paths) //得到文件路徑
 Assembly asm = Assembly.LoadFile(@"C:\Users\chy\Desktop\textdll\ToLower.dll");//程序集
 foreach (Type type in asm.GetTypes())
 { Console.WriteLine(type); }//程序集(命名空間)中的各種類
 Type t = asm.GetTypes()[0];
 Object obj = asm.CreateInstance(t.ToString());//創建類的實例(生成類的對象)
 MethodInfo m = t.GetMethods()[1];//得到指定類中的方法
 //MethodInfo m = t.GetMethod("方法名稱");
 news = (string)m.Invoke(obj, new Object[] { (Object)"aaBB" });//調用方法、如果是靜態方法則不用對象,也就是obj爲null
 Console.WriteLine(news);          
 MethodInfo mm = t.GetMethods()[0];
 Console.WriteLine(mm.Invoke(obj, null));


當然m文件生成的.Net Assembly dll會有好多默認的函數,自己可以通過getmethod輸出函數原型,經試驗,如果一個m文件其中就是一個函數的話,那麼生成的dll的函數一般如下

Void Dispose()
Void m1014094528rgbtogray()
Void m1014094528rgbtogray(MathWorks.MATLAB.NET.Arrays.MWArray)
MathWorks.MATLAB.NET.Arrays.MWArray[] m1014094528rgbtogray(Int32)
MathWorks.MATLAB.NET.Arrays.MWArray[] m1014094528rgbtogray(Int32, MathWorks.MATL
AB.NET.Arrays.MWArray)
Void WaitForFiguresToDie()
System.String ToString()
Boolean Equals(System.Object)
Int32 GetHashCode()
System.Type GetType()

其中m1014094528rgbtogray爲函數名,當然筆者寫的m文件只是簡單的將一幅圖像進行處理,沒有輸出參數,只有輸入參數,就是文件名。這裏便是使用Void m1014094528rgbtogray(MathWorks.MATLAB.NET.Arrays.MWArray)的函數,當然在使用時要添加MWArray的引用,安裝matlab安裝目錄下的..Matlab\toolbox\compiler\deploy\win32\MCRInstaller.exe,然後添加引用..\Matlab\toolbox\dotnetbuilder\bin\win32\v2.0\MWArray.dll,然後再在C#中using MathWorks.MATLAB.NET.Arrays;調用方法語句如下所示:m[2].Invoke(obj, new Object[] { (Object)(MathWorks.MATLAB.NET.Arrays.MWArray)@"D:\1.jpg"});也就是將文件名轉換爲MWArray的類型以供matlab調用,程序正確運行,結果也在意料之內。


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