C ++从struct数组得到错误的值

时间:2017-01-16 09:32:58

标签: c++ c arrays struct

问题如上所述。当我尝试从加载的* .so文件中读取值时(使用libdl),这些在struct中我得到了错误的值 申请代码:

#include <dlfcn.h>
#include <iostream>

/* For face data type reproduction */
#define GET_FACE_XYZ_SIZE 1
/* For face_array reproduction */
#define GET_FACE_ARRAY_SIZE 2
#define GET_OBJECT_DATA 3

typedef struct face {
    float x[1000];
    float y[1000];
    float z[1000];
    int vertices;
} face;


int main()
{
    void *hook;
    int (*fn)(int request_type, void *ptr);
    hook = dlopen("/root/osms/dlopen-test/lib.so", RTLD_LAZY);
    if(!hook)
    {
        std::cout << "Couldn't find lib.so" << std::endl;
    }
    fn = dlsym(hook, "object_info");
    int face_array_size = fn(GET_FACE_ARRAY_SIZE, NULL);
    std::cout << "FACE_ARRAY_SIZE: " << face_array_size << std::endl;
    face pointer[face_array_size];
    fn(NULL, pointer);
    dlclose(hook);
    std::cout << "pointer[0].z[1]: " << pointer[0].z[1] << std::endl;
return 0;
}

和lib.so的代码:

/* For face data type reproduction */
#define GET_FACE_XYZ_SIZE 1
/* For face array reproduction */
#define GET_FACE_ARRAY_SIZE 2
#define GET_OBJECT_DATA 3

typedef struct face {
    float x[1000];
    float y[1000];
    float z[1000];
    int vertices;
} face;

extern "C" int object_info(int request, void *ptr)
{
    face face_array[2];
    face_array[0].x[0] = 1.1;
    face_array[0].y[0] = 0.5;
    face_array[0].z[0] = 1.2;
    face_array[0].x[1] = 1.6;
    face_array[0].y[1] = -0.11;
    face_array[0].z[1] = -12;
    face_array[0].x[2] = -0.12;
    face_array[0].y[2] = 0.24;
    face_array[0].z[2] = -0.12;
    face_array[0].vertices = 3;

    face_array[1].x[0] = -1.1;
    face_array[1].y[0] = 0.15;
    face_array[1].z[0] = -1.2;
    face_array[1].x[1] = -1.6;
    face_array[1].y[1] = 0.11;
    face_array[1].z[1] = 1.2;
    face_array[1].x[2] = 0.12;
    face_array[1].y[2] = -0.24;
    face_array[1].z[2] = 0.12;
    face_array[1].vertices = 3;

    if(request == GET_FACE_ARRAY_SIZE)
    {
         return 2;
    }
    else
    {
         ptr = face_array;
    }
}

预期输出为pointer[0].z[1]: -12,但我得到pointer[0].z[1]: -0.12。我的代码有什么问题?

提前致谢

2 个答案:

答案 0 :(得分:1)

访问

pointer[0].z[1]

具有未定义的行为,因为它具有不确定的值。

object_info永远不会修改ptr指向的数组。它只是修改一个本地数组,并指定本地ptr指向该本地数组。

解决方案:不要声明本地数组,而是修改参数指向的数组。换句话说,使用以下内容重新face face_array[2];

face* face_array = (face*)ptr;

摆脱没有任何意义的ptr = face_array;

声明

object_info返回int,但并非所有代码路径都返回一个值。当函数在没有return语句的情况下到达object_info的末尾时,行为是未定义的。

解决方案:如果函数无效,则始终返回值。

face_array_size不是编译时常量值,因此face pointer[face_array_size];将声明一个可变长度数组。 C ++中不允许使用VLA。

要么使用C(自C99起支持VLA,但只有C11支持可选),或者使用动态数组:std::vector<face>或者说你的程序不符合标准这一事实。

答案 1 :(得分:0)

变量&#34; face_array&#34;在函数object_info和变量&#34;指针&#34;在main中不是同一个变量。 声明&#34; ptr = face_array&#34;不会改变&#34;指针&#34;的内容。

extern "C" int object_info(int request, face *face_array)
{
if(request == GET_FACE_ARRAY_SIZE)
    return 2;
face_array[0].x[0] = 1.1;
face_array[0].y[0] = 0.5;
face_array[0].z[0] = 1.2;
face_array[0].x[1] = 1.6;
face_array[0].y[1] = -0.11;
face_array[0].z[1] = -12;
face_array[0].x[2] = -0.12;
face_array[0].y[2] = 0.24;
face_array[0].z[2] = -0.12;
face_array[0].vertices = 3;

face_array[1].x[0] = -1.1;
face_array[1].y[0] = 0.15;
face_array[1].z[0] = -1.2;
face_array[1].x[1] = -1.6;
face_array[1].y[1] = 0.11;
face_array[1].z[1] = 1.2;
face_array[1].x[2] = 0.12;
face_array[1].y[2] = -0.24;
face_array[1].z[2] = 0.12;
face_array[1].vertices = 3;

}