c++ - 读取 mex 文件中的结构数据时发生奇怪的事情

标签 c++ c matlab structure mex

我刚才被一个很奇怪的mex错误搞糊涂了。 . .

将我的问题归结为核心,我们最终得到以下简单的 mex 代码。它只显示给定的结构字段是否为空 ...

#include "mex.h"

void mexFunction(int nlhs, mxArray *plhs[], int nrhs, const mxArray *prhs[])
{    
    int numElements  = mxGetNumberOfElements(prhs[0]);
    int numFields = mxGetNumberOfFields(prhs[0]);

    mxArray* tmpData;
    const char* tmpName;

    for (int structIdx=0; structIdx<numElements; ++structIdx)
    {
        for (int fieldIdx=0; fieldIdx<numFields; ++fieldIdx)
        {
            tmpData = mxGetFieldByNumber(prhs[0], structIdx, fieldIdx);
            tmpName = mxGetFieldNameByNumber(prhs[0], fieldIdx);

            if (mxIsEmpty(tmpData))
                mexPrintf("struct(%i).%s is empty\n", structIdx+1, tmpName );
            else
                mexPrintf("struct(%i).%s contains data\n", structIdx+1, tmpName );
        }
    }    
}

如果我们编译这段代码并将其命名为 structcrash 那么下面的 matlab 代码。 .

clc
x.a=1;
x.b=2;
x(2).a=3;
x(2).b=4;

structcrash(x);

...给出我们可能期望的输出...

  • struct(1).a 包含数据
  • struct(1).b 包含数据
  • struct(2).a 包含数据
  • struct(2).b 包含数据

如果我们给 mex 函数一个包含空字段的结构,就像这样......

clc
y.a = [];
structcrash(y);

...然后我们也得到预期的输出...

  • struct(1).a 为空

现在,如果你使用这样的代码,事情会变得很奇怪......

clc
y(2).b = 4;
structcrash(y);

如果我们检查 y 结构,它现在是一个 2 元素结构,每个元素中有 2 个字段。 y(1).a 正如我们上面指定的那样是空的,并且 y(1).b 在我们添加 时自动创建并赋予一个空值b 字段。类似地,当我们通过添加 y(2).b 来增加结构大小时,会自动创建 y(2).a。该结构看起来非常合乎逻辑,但是用作 mex 文件的输入会导致段错误。

通过有选择地注释掉多行代码,我可以确认导致段错误的命令是 mxIsEmpty(tmpData)

其他人可以复制这个错误吗?我在这里做错了什么吗?对我来说,它看起来像是 mex API 代码中的错误,但我想先在这里检查一下。谢谢

编辑:根据@David Heffernan 的建议,我修改了代码如下

        if(tmpData!=NULL) {
            if (mxIsEmpty(tmpData))
                mexPrintf("struct(%i).%s is empty\n", structIdx+1, tmpName );
            else
                mexPrintf("struct(%i).%s contains data\n", structIdx+1, tmpName );
        }

...并且段错误不再发生。不过,这还是很不祥的。如果您像下面的示例一样创建两个结构并使用工作区 View 检查它们,fg 看起来在各个方面都完全相同。使用标准的 matlab 编程命令,我找不到任何不同之处。

>> f(2).a=123;
>> g(1).a=[];
>> g(2).a=123

...但是 whos 命令揭示了不同...

  Name      Size            Bytes  Class     Attributes

  f         1x2               192  struct              
  g         1x2               296  struct 

...我更新的 mex 函数显然也是...

>>structcrash(f)
struct(2).a contains data
>> structcrash(g)
struct(1).a is empty
struct(2).a contains data

所以这个故事的寓意是,当您将新字段插入特定结构元素时,Matlab IDE 通过在所有结构中插入字段来使结构看起来漂亮且方正。然而,实际上,在底层内存中,情况并非如此。

小心!

最佳答案

发生的事情是 mxGetFieldByNumber 返回 NULL,然后您将其传递给 mxIsEmpty 并因此产生段错误。文档指出,如果没有为指定字段分配值,则 mxGetFieldByNumber 返回 NULL

要解决这个问题,您需要防止将 NULL 传递给 mxIsEmpty:

if (tmpData == NULL || mxIsEmpty(tmpData))
    mexPrintf("struct(%i).%s is empty\n", structIdx+1, tmpName);
else
    mexPrintf("struct(%i).%s contains data\n", structIdx+1, tmpName);

关于c++ - 读取 mex 文件中的结构数据时发生奇怪的事情,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/10706739/

相关文章:

Matlab:如何在没有 Tab 的情况下激活自动完成?

c++ - 使用枚举处理错误

c++ - 预分配列表

c++ - 拦截 VBA 应用程序之间的消息

python - 从 numpy 矩阵获取项目,索引位于数组中

matlab - 使用 surf 创建具有透明背景的 png 文件的 3D 图像

c++ - 指向不同 .h 文件的类指针

c - sizeof(1) 是多少,它与 1 有什么不同?

不能从 for 循环中的 float 中减去浮点值?

c - C 编程语言,第 1 章练习 1.10(Getchar 和 Putchar)