Matlab与C#连接的几种方式比较

http://www.cnblogs.com/ceachy/articles/2153322.html

使用环境 Visual Studio 2005,Matlab 2007a。

前提:机器要装好MCR(很变态,100MB~200MB因版本而异),否则会编译出错。

1.COM

步骤:

matlab编译工作 
- mbuild -setup 
- deploytool,Matlab Builder for .net,Generic COM Component 
- 添加m函数文件(eg: myfunc.m),改类名(eg: myComClass),build

注册dll 
regsvr32 myCom.dll 
regsvr32 mwcomutil.dll

VS2005调用: 
- Reference加Com控件 
- 调用示例 
double[,] arr = null; 
object in_a, out_a; 
in_a = 500; 
out_a = arr1; 
myCom.myComClass mc = new myCom.myComClass(); 
mc.myfunc(1, ref out_a, in_a);

注: 
- 如果报错,且报错中有乱码,请自己打开log看个究竟,log里面可显示出中文。 
- VS2005必须装上VC,否则找不到Midl这个文件,就编译不了com

2. .net(参见Matlab Builder for .net的帮助,有源代码)

matlab 2006a版本之后提供此连接方式。

步骤:

matlab编译工作 
- mbuild -setup 
- deploytool,Matlab Builder for .net,.NET Component 
- 添加m函数文件(eg: myfunc.m),改类名(eg: myClass),build 
VS2005调用: 
- Reference要有MWArray(%matlabpath%\toolbox\dotnetbuilder\bin\win32\v2.0)和build出来的DLL 
- 使用名空间 
using MathWorks.MATLAB.NET.Utility; 
using MathWorks.MATLAB.NET.Arrays; 
using dll名; 
- MWNumericArray是MWArray和c#中数据的中间类,怎么用?怎样在C#与matlab间传递参数? 
a.double型、int型等数值类型的变量传递 
MWNumericArray i = null, result = mydouble; 
i=4; 
myClass myclass = new myClass(); //实例化 
result = (MWNumericArray)myclass.myfunc(i);

b.字符串(需要用到MWCharArray和MWArray转换) 
MWCharArray FileName = myString; 
MWNumericArray sensitivity; 
sensitivity = (MWNumericArray)myClass.myAlgorithm((MWArray)FileName);

c.多个输出参数组成的数组 
MWNumericArray out_Arr = (MWNumericArray)out_Args[1]; //取出第一个参数返回的数组(matlab返回的数组下界是从1开始的)

取出数组中的一个元素值 
mydouble=out_Arr[i].ToScalarDouble();

如上面例子ToScalarDouble一类的toXXXX等方法还有许多,慢慢发掘吧。

d.result.toArray可以把matlab返回的矩阵变成C#的n x m数组,eg:

double[,] csArray= (double[,])result.ToArray(MWArrayComponent.Real);

e.将数组传入Matlab(和传递单个数值的方法一样) //4月26日添加 
double[,] dbx = new double[2, 2] { { 1, 2 }, { 3, 4 } }; 
MWNumericArray x=dbx; 
myclass.picture(x);

其中picture是一个自己写的m函数,内容是plot(x),用于验证传入的矩阵的结果。结果如下图:

3.时间、CPU、内存开销比较:

时间是这样算得的: 
DateTime tst0 = DateTime.Now; 
... 
DateTime tst1 = DateTime.Now; 
TimeSpan dift = tst1 - tst0;

Matlab-C#内存/虚拟内存开销(MByte) 
----------------------------------- 
1个magic(4) 77.9, 158.4 
2个magic(4) 77.9, 158.4 
1个magic(500) 84.7, 165.2 
无界面程序 7.4M, 8M

时间测试(s) .NET / COM 
-----------------------------------Group1 
实例化时间 3.2813 3.2813 
第一次调magic(4) 0.0469 0.0625 
第二次调magic(5) 0.0000 0.0313 
传递5x5矩阵 0.1875 未测 
-----------------------------------Group2 
实例化时间 3.2656 3.3125 
第一次调magic(500) 0.1406 0.1719 
第二次调magic(557) 0.0156 0.1250 
传递500x500矩阵 0.2969 未测 
-----------------------------------Group3 
实例化时间 3.3125 未测 
第一次调magic(4) 0.0469 未测 
第二次调sumab(1,1) 0.0156 未测

结论: 
- 开机后第一次运行程序,需要花10s左右的时间实例化类,之后实例化需要花费3s的时间。 
- 耗内存、占时间的是将MCR实例化的过程!! 
- 最好在程序靠前位置把封装的类实例化。 
- 大矩阵的类型转换要消耗一段时间,但是和小矩阵比并不明显。如果数据量实在很大,用文件传递参数也是一个解决办法。 
- com模式和.Net模式的调用相差不大,但就编译速度而言.net的方式要快上很多,而且似乎比较稳定。(我的机器就是如此,com编译报错但.net编译很快就通过了,而且用得不错) 
- 早期版本的MCR库较小,估计能快一些。 
- 有的机器无法编译Com,可能是Windows XP缺乏补丁。据说“番茄花园”等版本系统的一些服务被“优化”掉了,也不能够正常编译。 
- 更新matlab程序,将dll复制到相应文件夹后,应该删除文件夹下的(dll名字)_mcr文件夹,否则程序在载入dll时可能出现异常。

----------------------------

调试经验(2009年3月30日更新)

由于matlab对输入参数的处理非常灵活,而C#在输入输出上十分严格,因此常会出现转换数据类型出错的问题。

解决方法是把C#接口的输入输出看牢:Matlab函数的输入参数最好为double型,不要double和float混杂,否则输出可能有些是float有些是double.

-------------------------------

关于数据接口的两种处理方法(2009年7月22日更新)

上面的时间测试表格给出的是值传递方法消耗的时间。实际使用中发现传递参数很多时,值传递法需要消耗大量时间。因此又提出了通过文件传递参数或结果的方法。以下是在上述两种方法下,运算与返回数据消耗的时间评估。

时间测试(s) 值传递 / 文件 
-----------------------------------Group1 
实例化时间 3.5625 3.5937 
第一次调magic(4) 0.0469 0.0781 
第二次调magic(500) 0.1250 0.1406 
传递500x500矩阵 0.2812 0.0000(无法测出)

由于文件传递500x500的整形矩阵时间太短无法测出,故改为800x800以评估其时间。消耗时间为:0.0156

时间: 2024-11-04 23:52:23

Matlab与C#连接的几种方式比较的相关文章

python字符串连接的N种方式

python中有很多字符串连接方式,今天在写代码,顺便总结一下: 最原始的字符串连接方式:str1 + str2 python 新字符串连接语法:str1, str2 奇怪的字符串方式:str1 str2 % 连接字符串:‘name:%s; sex: ’ % ('tom', 'male') 字符串列表连接:str.join(some_list) 第一种,想必只要是有编程经验的人,估计都知道,直接用 “+” 来连接两个字符串: 'Jim' + 'Green' = 'JimGreen' 第二种比较特

Java字符串连接的几种方式

Java字符串连接的几种方式 字符串表现的几种方式 StringBuffer和StringBuilder及String的继承关系 字符串的连接 1.String的连接方法 可以看出连接方式是新建了一个包含两个长度的字符数组,然后进行连接. 2.StringBuilder中存储字符串其实用的是一个char数组,capacity其实就是指定这个char数组的大小,StringBuilder的连接方法是继承AbstractStringBuilder的方法的,线程不安全的 在append(str)函数调

多表连接的三种方式详解 HASH JOIN MERGE JOIN NESTED LOOP

在多表联合查询的时候,如果我们查看它的执行计划,就会发现里面有多表之间的连接方式. 之前打算在sqlplus中用执行计划的,但是格式看起来有点乱,就用Toad 做了3个截图. 从3张图里我们看到了几点信息: 1.       CBO 使用的ALL_ROWS模式 Oracle Optimizer CBO RBO http://blog.csdn.NET/tianlesoftware/archive/2010/08/19/5824886.aspx 2.       表之间的连接用了hash Join

C#连接mysql三种方式

第一种方式: 使用MySQLDriverCS.dll连接 MySQLDriverCS软件下载:http://sourceforge.net/projects/mysqldrivercs/?source=typ_redirect 安装完之后再引用中添加引用,找到安装目录,找到MySQLDriverCS.dll文件,然后添加using MySQLDriverCS.dll文件 参考网址:http://www.cnblogs.com/genli/articles/1956537.html C#连接mys

sql表连接的几种方式

这里有两张表TableA和TableB,分别是姓名表和年龄表,用于我们例子的测试数据 TableA id name 1 t1 2 t2 4 t4 TableB id age 1 18 2 20 3 19 在开发中我们的业务需求有时候是复杂的,多张表联合查询的时候是有多种方式的,面对不同的需求, 灵活使用不同的表连接方式,那么表连接分成哪几种呢? 表连接有几种? sql表连接分成外连接.内连接和交叉连接. 一.外连接 概述: 外连接包括三种,分别是左外连接.右外连接.全外连接. 对应的sql关键字

虚拟机 linux 网络连接的三种方式

在VMare虚拟机中,网络连接有三种方式 1.桥接模式   2.only host模式     3.NAT模式   当然还可以自定义,但是虚拟机提供的只有三种方式. 在装完虚拟机之后,网络适配器中就会多出来两个虚拟网卡,VMent1 和Vment8.在使用桥接模式的时候是不需要使用虚拟网卡的,因为它使用的是 你的真实网卡,也就是有线网卡,所以有时候使用桥接模式链接的时候会发现自己没有连通,原因可能是使用了无线网卡,在虚拟机桥接模式中将自动链接 改成自己的有线网卡就可以了. 使用桥接模式因为使用的

HBase连接的几种方式(二)

1. HBase连接的方式概况 主要分为: 纯Java API连接HBase的方式: Spark连接HBase的方式: Flink连接HBase的方式: HBase通过Phoenix连接的方式: 第一种方式是HBase自身提供的比较原始的高效操作方式,而第二.第三则分别是Spark.Flink集成HBase的方式,最后一种是第三方插件Phoenix集成的JDBC方式,Phoenix集成的JDBC操作方式也能在Spark.Flink中调用. 注意: 这里我们使用HBase2.1.2版本,以下代码都

进行蓝牙连接的两种方式

为了在两台设备间创建一个连接,必须实现服务器端和客户端的机制,因为一个设备必须打开一个Server Socket,而另一个必须发起连接(使用服务器端设备的MAC地址发起连接).当服务器端和客户端在同一个RFCOMM信道上都有一个BluetoothSocket时,则两端就建立了连接.此刻,每个设备都能获得一个输入输出流,进行数据传输.服务器端和客户端获得BluetoothSocket的方法是不同的,服务器端是在客户端的连接被接受时才产生一个BluetoothSocket,客户端是在打开一个到服务器

C#与数据库的连接的三种方式

学习了.net的知识从C#一直到MVC,我一直觉得基础很重要,最近有复习一下数据库连接的三种方式 1 返回结果集的一张表 public static DataTable ExecuteDataTable(string sql, params SqlParameter[] parameters) { DataSet ds = new DataSet(); SqlDataAdapter adapter = new SqlDataAdapter(sql, str); adapter.SelectCom