在mex文件中读取结构数据时会发生奇怪的事情

时间:2012-05-22 17:07:27

标签: 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).a来增加结构大小时,会自动创建y(2).b。该结构看起来非常合乎逻辑,但是使用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 );
        }

...并且不再发生段错误。然而,这仍然是非常不祥的。如果您创建两个结构(如以下示例中所示)并使用工作区视图检查它们,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通过在所有结构中插入字段使结构看起来很漂亮。但是,实际上,在底层内存中,情况并非如此。

小心!

1 个答案:

答案 0 :(得分:2)

正在发生的事情是mxGetFieldByNumber正在返回NULL,然后您将其传递给mxIsEmpty,从而产生seg错误。该文档指出,如果没有为指定字段分配值,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);