从 C++ 调用 MATLAB 错误:无法解析的外部符号

问题描述 投票:0回答:2

从 C++ 调用 MATLAB 函数时遇到几个错误。主要思想是:首先编译一个MATLAB函数并生成DLL文件,然后在C++中包含.h和.lib文件。最后编写.cpp来测试并调用该函数。这是我的详细步骤,请告诉我哪里错了。

(使用 MATLAB 2012b 和 Visual C++ 2008,Windows 7 64 位)

在 MATLAB 中:

  1. mbuild -setup
    mex -setup
    将 Visual Microsoft Visual C++ 2008 SP1 设置为编译器。
  2. 在文件夹

    MyAdd.m
    中创建
    C:\Users\WangYudong\Documents\MATLAB\MyAdd_M
    ,功能如下:

    function [c] = MyAdd(a, b)
    c = a + b;
    
  3. mcc -W cpplib:libMyAdd -T link:lib MyAdd
    编译
    MyAdd.m
    并生成多个文件,包括
    libMyAdd.dll
    libMyAdd.h
    libMyAdd.lib
    等文件。

在 C++ 中

  1. 选择 VC++ 目录包含文件 添加

    E:\MATLAB\R2012b\extern\include

  2. 选择VC++目录库文件添加

    E:\MATLAB\R2012b\extern\lib\win64\microsoft
    C:\Users\WangYudong\Documents\MATLAB\MyAdd_M

  3. 选择链接器输入附加依赖项添加新条目:

    mclmcr.lib
    mclmcrrt.lib
    libmx.lib
    libmat.lib
    libMyAdd.lib
    
  4. 创建一个新的

    MyAdd_test.cpp
    并将
    libMyAdd.dll
    libMyAdd.h
    libMyAdd.lib
    放在同一文件夹中。在
    头文件
    中添加libMyAdd.h,在
    资源文件
    中添加
    libMyAdd.h
    libMyAdd.lib

MyAdd_test.cpp
的代码如下:

#include "mclmcr.h"
#include "matrix.h"
#include "mclcppclass.h"
#include "libMyAdd.h"

int main() {   
    double a = 6;
    double b = 9;
    double c;
    // initialize lib
    if( !libMyAddInitialize()) {
        std::cout << "Could not initialize libMyAdd!" << std::endl;
        return -1;
    }

    // allocate space
    mwArray mwA(1, 1, mxDOUBLE_CLASS);
    mwArray mwB(1, 1, mxDOUBLE_CLASS);
    mwArray mwC(1, 1, mxDOUBLE_CLASS);  
    // set data
    mwA.SetData(&a, 1);
    mwB.SetData(&b, 1);
    // use function: MyAdd
    MyAdd(1, mwC, mwA, mwB);
    // get data
    c = mwC.Get(1, 1);
    printf("c is %f\n", c); 

    // terminate the lib
    libMyAddTerminate();   
    // terminate MCR
    mclTerminateApplication();
    return 0;
}  

最后的结果是

Compiling...
MyAdd_test.cpp
Linking...
MyAdd_test.obj : error LNK2019: unresolved external symbol _mclTerminateApplication_proxy referenced in function _main
MyAdd_test.obj : error LNK2019: unresolved external symbol _libMyAddTerminate referenced in function _main
MyAdd_test.obj : error LNK2019: unresolved external symbol "__declspec(dllimport) void __cdecl MyAdd(int,class mwArray &,class mwArray const &,class mwArray const &)" (__imp_?MyAdd@@YAXHAAVmwArray@@ABV1@1@Z) referenced in function _main
MyAdd_test.obj : error LNK2019: unresolved external symbol _libMyAddInitialize referenced in function _main
MyAdd_test.obj : error LNK2019: unresolved external symbol _mclGetMatrix referenced in function "public: __thiscall mwArray::mwArray(unsigned int,unsigned int,enum mxClassID,enum mxComplexity)" (??0mwArray@@QAE@IIW4mxClassID@@W4mxComplexity@@@Z)
MyAdd_test.obj : error LNK2019: unresolved external symbol _mclcppGetLastError referenced in function "public: static void __cdecl mwException::raise_error(void)" (?raise_error@mwException@@SAXXZ)
MyAdd_test.obj : error LNK2019: unresolved external symbol _mclcppCreateError referenced in function "public: __thiscall mwException::mwException(void)" (??0mwException@@QAE@XZ)
MyAdd_test.obj : error LNK2019: unresolved external symbol _ref_count_obj_addref referenced in function "public: __thiscall mwException::mwException(class mwException const &)" (??0mwException@@QAE@ABV0@@Z)
MyAdd_test.obj : error LNK2019: unresolved external symbol _ref_count_obj_release referenced in function "public: virtual __thiscall mwException::~mwException(void)" (??1mwException@@UAE@XZ)
MyAdd_test.obj : error LNK2019: unresolved external symbol _error_info_get_message referenced in function "public: virtual char const * __thiscall mwException::what(void)const " (?what@mwException@@UBEPBDXZ)
MyAdd_test.obj : error LNK2019: unresolved external symbol _array_ref_getV_int referenced in function "public: class mwArray __cdecl mwArray::GetPromoted(unsigned int,...)" (?GetPromoted@mwArray@@QAA?AV1@IZZ)
MyAdd_test.obj : error LNK2019: unresolved external symbol _array_ref_set_numeric_mxDouble referenced in function "public: void __thiscall mwArray::SetData(double *,unsigned int)" (?SetData@mwArray@@QAEXPANI@Z)
MyAdd_test.obj : error LNK2019: unresolved external symbol _array_ref_get_numeric_mxDouble referenced in function "public: __thiscall mwArray::operator double(void)const " (??BmwArray@@QBENXZ)
C:\Users\WangYudong\Documents\Visual Studio 2008\Projects\MyAdd_C\Debug\MyAdd_C.exe : fatal error LNK1120: 13 unresolved externals

实际上,上面的工作是我从 C++ 调用自定义 MATLAB 函数的测试。我接下来的工作是将MATLAB程序转换为C++,其中包含图像处理函数,如

imread
edge
strel
等。我尝试过MATLAB Coder,但它无法转换MATLAB函数。所以我尝试上面的方法。 这是转换这些函数的有效方法吗还是我应该使用 OpenCV 来实现它们?

c++ matlab matlab-compiler
2个回答
6
投票

开始之前,请确保安装了支持的编译器。如果您在 64 位 Windows 上使用 VS Express 版本,则可能是 Visual C++,也可能是 Windows SDK。接下来,您需要通过运行这些步骤至少一次来配置 MATLAB:

>> mex -setup
>> mbuild -setup

现在给出以下简单函数:

我的添加.m

function c = MyAdd(a,b)
    c = a + b;
end

我们想要使用 MATLAB 编译器mcc构建一个

C++ 共享库
:

>> mcc -N -W cpplib:libMyAdd -T link:lib MyAdd.m -v

这将生成几个文件,包括头文件、DLL 和导入库:

libMyAdd.h
libMyAdd.dll
libMyAdd.lib

接下来我们创建一个C++程序来测试上面的库:

MyAdd_test.cpp

#include "libMyAdd.h"

int main()
{
    // initialize MCR and lib
    if (!mclInitializeApplication(NULL,0))  {
        std::cerr << "could not initialize the application" << std::endl;
        return -1;
    }
    if(!libMyAddInitialize()) {
        std::cerr << "Could not initialize the library" << std::endl;
        return -1;
    }

    try {
        // create input
        double a[] = {1.0, 2.0, 3.0, 4.0};
        double b[] = {5.0, 6.0, 7.0, 8.0};
        mwArray in1(2, 2, mxDOUBLE_CLASS, mxREAL);
        mwArray in2(2, 2, mxDOUBLE_CLASS, mxREAL);
        in1.SetData(a, 4);
        in2.SetData(b, 4);

        // call function
        mwArray out;
        MyAdd(1, out, in1, in2);

        // show result
        std::cout << "in1 + in2 = " << std::endl;
        std::cout << out << std::endl;

        double c[4];
        out.GetData(c, 4);
        for(int i=0; i<4; i++) {
            std::cout << c[i] << " " << std::endl;
        }

    } catch (const mwException& e) {
        std::cerr << e.what() << std::endl;
        return -2;
    } catch (...) {
        std::cerr << "Unexpected error thrown" << std::endl;
        return -3;
    } 

    // cleanup
    libMyAddTerminate();   
    mclTerminateApplication();

    return 0;
}

我们可以使用以下方法直接从 MATLAB 内部编译该程序:

>> mbuild MyAdd_test.cpp libMyAdd.lib -v
>> !MyAdd_test

我们也可以使用 Visual Studio 自己编译它。我们首先创建一个本机控制台应用程序,然后按如下方式设置项目设置:

  • 从菜单中选择“项目 > 属性”并将设置应用到“所有配置”(调试和发布目标)
  • 在 C/C++ 属性下,除了包含 MATLAB 头文件的目录之外,还添加包含生成的头文件的目录

    libMyAdd.h
    来设置“附加包含目录”:

    $matlabroot\extern\include
    
  • 同样在“链接器”下,设置“附加库目录”。这将与之前包含

    libMyAdd.lib
    的目录相同,在我的情况下也是如此:

    $matlabroot\extern\lib\win32\microsoft
    

    然后转到“链接器 > 输入”并在“附加依赖项”中添加以下内容:

    libMyAdd.lib
    mclmcrrt.lib
    
  • 最后,在“调试 > 环境”下,您可能需要扩展

    PATH
    环境变量以包含包含生成的
    libMyAdd.dll
    文件的目录。这样你就可以直接按F5从VS内部直接编译运行程序。这将类似于:

    PATH=%PATH%;C:\path\to\output\folder
    

如果你经常做这种事情,你可以创建一个属性表一次,然后可以在其他 VC++ 项目中重复使用。请参阅此 answer 作为示例。


0
投票

我使用以下命令之一在 Matlab 中编译了 C++ 库,并将 Matlab 配置为使用 MingGW。

mcc -W cpplib:fooLIb FooFunc.m
mcc -B cpplib:fooLIb FooFunc.m

我尝试将此库与 Visual Studio 项目一起使用,并得到一个未解析的外部符号。解决方案是将 Matlab 配置为使用 Visual Studio 编译器。要查看默认编译器,请运行下面的命令(两者都很重要)。单击提供的链接设置默认编译器。

mex -setup cpp
mbuild -setup cpp

值得注意的是,使用 MinGW 编译的 C 库即使在 Visual Studio C++ 项目中也能正常工作。

© www.soinside.com 2019 - 2024. All rights reserved.