13

您可以像这样在 C 中声明一个结构:

typedef struct MyStruct {
    const char *name;
    int (*func1)(void);
    int (*func2)(void);
    int (*func3)(void);
} MyStruct;

int test_func2(void) {
    return 0;
}

MyStruct test_struct = {
    .name   = "buffer",
    .func2  = test_func2,
};

这对于仅定义特定成员非常方便,所有其他成员都设置为 0/NULL。

编辑:特别是,这允许不知道如何定义 MyStruct 的详细信息,因此它可以在内部更改、添加新成员等。而无需使用这种类型破坏代码。

然而,这不会用 C++ 编译器编译,出现错误:

test.c:23: error: expected primary-expression before ‘.’ token
test.c:24: error: expected primary-expression before ‘.’ token

是否有实现相同的等效 C++ 声明?

谢谢。

编辑:@chris 我可以告诉你不明白 :) 而且很明显,大多数其他人评论我应该使用什么语法,应该如何定义结构等等。完全没有抓住重点。这不是关于定义结构的正确方法,这个片段只是为了提供一个上下文。

至于代码等效性,请在您的代码中某处说:

MyStruct blah = { NULL, NULL, func2 };

现在 MyStruct 将其定义更改为:

typedef struct MyStruct {
  const char *name;
  int (*func4)(void);
  int (*func1)(void);
  int (*func2)(void);
  int (*func3)(void);
} MyStruct;

您的代码仍然可以正常编译,但引入了严重的回归:您现在将初始化 func1 成员,而不是像以前那样设置 func2...

问题是关于 C++ 指定的初始值设定项是否等效:没有。问题关闭。

4

4 回答 4

16

不,C++ 不支持 C99 的指定初始化程序。如果要按名称设置单个成员,则需要通过分配来完成,例如

MyStruct test_struct = MyStruct();
test_struct.name  = "buffer";
test_struct.func1 = test_func1;
于 2012-06-02T01:51:28.820 回答
2

正如 K-ballo 所说,代码相当于:

MyStruct test_struct = {
    "buffer",
    test_func1
};

但是由于您使用的是 C++,因此您应该考虑为 MyStruct 编写和使用构造函数:

struct MyStruct {
    const char *name;
    int (*func1)(void);
    int (*func2)(void);
    int (*func3)(void);

    MyStruct(const char *n,
        int (*f1)(void) = NULL,
        int (*f2)(void) = NULL,
        int (*f3)(void) = NULL) : name(n), func1(f1), func2(f2), func3(f3) {}
};

请注意,在 C++ 中,您不需要将其 typedef 到 MyStruct,即使没有 typedef,您也可以只使用 MyStruct(没有 struct 关键字)。

于 2012-06-02T01:58:13.747 回答
2

您可以只使用 lambda(自 C++11 起)。这将允许您保持声明不变,同时轻松初始化结构的全局/常量实例:

struct Test
{
    int x;
    float y;
};

const Test test=[]{ Test t{}; t.x=5, t.y=3.3465; return t; }();

#include <iostream>
int main()
{
    std::cout << "test.x: " << test.x << ", test.y: " << test.y << "\n";
}

你甚至可以将它包装到一个宏中,尽管在下面的表格中它只允许固定数量的初始化器:

// Macro to simplify it a bit
#define DECL_AND_INIT(Type,object,INIT1,INIT2) \
Type object=[]{ Type o{}; o.INIT1; o.INIT2; return o; }()
// Example usage
const DECL_AND_INIT(Test,test1,x=-7,y=-9.325e31);

请注意,您仍然必须像Test t{};在 lambda 中那样使用值初始化——否则,如果您没有初始化所有成员(例如,由于添加了新成员),您最终将复制导致 UB 的未初始化变量。

于 2016-09-27T11:23:44.843 回答
1

this 的功能等效性将是您struct的空主体的构造函数:

#ifdef __cplusplus
MyStruct::MyStruct(char* N, int (*F)(void)) :
    name(N),
    func1(0),
    func2(0),
    func3(F) {
  // empty
}
#endif

(以一些语法错误为模,我的 C++ 有点生疏。)

这在功能上是等效的,因为它允许您构造全局对象,即使它们具有const限定字段,并且任何现代编译器都应该在编译时使用静态链接初始化对象。

于 2012-06-02T08:24:33.713 回答