重庆分公司,新征程启航
为企业提供网站建设、域名注册、服务器等服务
运行环境:Windows 8 Release Preview(Build 8400),Matlab R2012a,Visual Studio 2010
超过10年行业经验,技术领先,服务至上的经营模式,全靠网络和口碑获得客户,为自己降低成本,也就是为客户降低成本。到目前业务范围包括了:成都网站制作、成都网站设计,成都网站推广,成都网站优化,整体网络托管,小程序设计,微信开发,成都app软件开发,同时也可以让客户的网站和网络营销和我们一样获得订单和生意!
把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调用,程序正确运行,结果也在意料之内。