【从业余项目中学习2】C# 实现调用Matlab函数(Visual Studio:2008, Matlab:R2009a)


  最近正在给客户做的个人项目,要求实现C#与Matlab之间的调用,即C# winform界面收集用户输入的参数,将参数传递给Matlab的算法计算,Matlab函数返回的结果显示在winform界面上。

  网上此类的文章较多,但自己在实现过程中还是有些差别,所以在项目进行之前,自己写了一个测试的例子来实现C#对Matlab函数的调用

一.  测试用例简介

  功能:Matlab函数计算两个数值a与b的和,a与b的值由C#提供,和值c经Matlab计算得出后,返回给C#

  环境:Microsoft Visual Studio 2008

     Matlab R2009a(Version 7.8.0.347)

  备注:由官网可知,Matlab对类似其他程序调用都提供了很好的支持,这里没有选择VS版本大于Matlab版本,是因为担心Matlab版本只支持自己之前的VS版本。

二.  实现步骤

  1. 实现Matlab函数

    A. 打开Matlab R2009a,新建testAdd.m文件

    B. testAdd.m中实现加法函数

1 function y = testAdd(a, b)
2 y = a + b;

  2. 编译testAdd.m,得到DLL文件,以便由C#引用

    A. 在Matlab命令行中输入"deploytool",即可弹出"Deployment Tool"工具窗口

      在网上看很多资料,执行"deploytool"命令之前都需要安装Matlab编译器(命令:"mbuild -setup"),但我不这样做,也可编译。怀疑是否与我将VS,Matlab都装在一台机器有关。另外即便我执行这安装命令,也找不到正确的编译器。总之,我并没有按照网上教程,直接"deploytool"即可。这也提醒自己:实践过后,才知是否正确,不要盲目听从别人方法。

    B. 编译DLL,需要在"Deployment Tool"工具窗口中新建Deployment project

      这里我建了名为"test"的工程,选择project类型时,应选择.NET Component,因为这里我需要其作为C#的引用。同时注意这里的工程名,即是你编译出DLL的名称,同时C#程序调用时,"Test"即为封装Matlab函数的类名。他将你的工程名,首字母大写用为高级语言中的类名。

    C. 配置"test" project

      首先将testAdd.m添加到test工程下Test文件夹中(右击Test,选择Add File)(注意:这里不要使用中文路径,详见下方六.测试过程中Bug记录)。

      其次选择Setting,在设置中,配置.NET Microsoft Framework,由"Default"改为"2.0",不能用默认。网上说法是否则编译出的DLL会有问题。

      注意:这里网上许多资料讲,要将Assembly Type设置为Shared,但我发现如果这样,必须要提供Encryption Key的文件路径,这个就没法提供了。所以我并没有这样做,事实上没有影响。疑惑的地方。

    D. 点击"Build"编译文件(如下图所示)

      

     即可在test\src路径下得到编译后的DLL文件

     

  3. 实现C#程序,调用Matlab编译出的DLL

    A. C#项目中,导入Matlab DLL引用

      导入的文件:test.dll, testNative.dll, MWArray.dll(%matlabpath%\toolbox\dotnetbuilder\bin\win32\v2.0,MWArray是用于C#与Matlab之间的数据交换类,传值,取结果都用到它)

    B. 实现C#调用代码

 1 using System.Data;
 2 using System.Drawing;
 3 using System.Linq;
 4 using System.Text;
 5 using System.Windows.Forms;
 6 using MathWorks.MATLAB.NET.Utility;
 7 using MathWorks.MATLAB.NET.Arrays;
 8 using test;
 9 
10 namespace testMatlab
11 {
12     public partial class Form1 : Form
13     {
14         public Form1()
15         {
16             InitializeComponent();
17 
18             label6.Text = "";
19         }
20         
21         private void button1_Click(object sender, EventArgs e)
22         {
23             //Get input number from UI
24             int iTextBox1 = int.Parse(textBox1.Text.ToString());
25             int iTextBox2 = int.Parse(textBox2.Text.ToString());
26                         
27             MWArray result = null;
28             MWNumericArray a = new MWNumericArray(iTextBox1);
29             MWNumericArray b = new MWNumericArray(iTextBox2);
30 
31             
32             //call function testAdd, provided by Matlab
33             Test t = new Test();
34             result = t.testAdd((MWArray)a, (MWArray)b);
35 
36 
37             int y = ((MWNumericArray)result).ToScalarInteger();
38             
39             //show result in UI
40 
41             textBox3.Text = y + "";
42             label6.Text = "Get the result by Matlab DLL, Answer: " + y;
43         }
44     }
45 }

三.  执行C# winform程序,验证计算结果,确定是否执行了Matlab函数调用

    1. 输入初值

   

    2. 得到结果

   

    

四.  小结

  由此,可实现C#对Matlab算法的调用。当然这里只是自己开发需要时,设计的一个小例子。由于Matlab计算,输入与结果需要大量的矩阵,那C#与其之间的数据交换也是通过MWArray进行,只要遵循接口规范,都可以实现。这里不再研究,网上资料也很多。

五.  完整的测试例子附件

  Matlab函数:https://files.cnblogs.com/KevinSong/testMatlab.zip

  C#程序:https://files.cnblogs.com/KevinSong/Test.zip

.  测试过程中Bug记录

  1. 调用DLL时,程序报错"MathWorks.MATLAB.NET.Utility.MWMCR.mclCreateComponentData,错误描述是:传递给系统调用的数据区域太小"

    解决方法:不能使用中文路径的.m文件,可能会有诡异的问题产生

    参考链接:http://hi.baidu.com/ssemo/item/4caab7f3a0765ec20dd1c888

虽然网上参考资料很多,但经过自己实践,还是发现一些不同的小地方。项目进行前,通过一个小的测试例子来证明技术路线可行。希望对需要的同学有帮助。抛砖引玉:-)

Best Regards

Kevin Song