.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中的任何类和方法,然后进行调用。

先写一个类:

View Code
 1 namespace ToLower
 2 {
 3     public class Tolower
 4     {
 5         public string exe()
 6         {
 7              return "tolower"; 
 8         }
 9 
10         public string execute(string s)
11         {
12             return s.ToLower();
13         }
14     }
15     public class tt
16     { }
17 }

然后通过反射获得类以及方法名,进而调用其方法。代码如下:

View Code
 1 string news;
 2 //string[] paths = Directory.GetFiles(@"C:\Users\chy\Desktop\textdll\", "*.dll");
 3 //foreach (string path in paths) //得到文件路径
 4 Assembly asm = Assembly.LoadFile(@"C:\Users\chy\Desktop\textdll\ToLower.dll");//程序集
 5 foreach (Type type in asm.GetTypes())
 6 { Console.WriteLine(type); }//程序集(命名空间)中的各种类
 7 Type t = asm.GetTypes()[0];
 8 Object obj = asm.CreateInstance(t.ToString());//创建类的实例(生成类的对象)
 9 MethodInfo m = t.GetMethods()[1];//得到指定类中的方法
10 //MethodInfo m = t.GetMethod("方法名称");
11 news = (string)m.Invoke(obj, new Object[] { (Object)"aaBB" });//调用方法、如果是静态方法则不用对象,也就是obj为null
12 Console.WriteLine(news);           
13 MethodInfo mm = t.GetMethods()[0];
14 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调用,程序正确运行,结果也在意料之内。

posted on 2012-10-13 14:25  月流霜  阅读(1533)  评论(0编辑  收藏  举报