我想初始化一个struct元素,在声明和初始化中分裂。这是我所拥有的:

typedef struct MY_TYPE {
  bool flag;
  short int value;
  double stuff;
} MY_TYPE;

void function(void) {
  MY_TYPE a;
  ...
  a = { true, 15, 0.123 }
}

这是根据C编程语言标准(C89、C90、C99、C11等)声明和初始化MY_TYPE局部变量的方法吗?或者有更好的方法或者至少有效的方法吗?

更新我最终有一个静态初始化元素,我设置每个子元素根据我的需要。


当前回答

C编程语言标准ISO/IEC 9899:1999(通常称为C99)允许使用指定的初始化式来初始化结构或联合的成员,如下所示:

MY_TYPE a = { .stuff = 0.456, .flag = true, .value = 123 };

ISO/IEC 9899:1999标准第6.7.8节初始化第7段定义为:

如果指示符具有该形式 . 标识符 那么当前对象(定义如下)将具有结构类型或联合类型,标识符将是该类型成员的名称。

注意,同一节第9段规定:

除非另有明确说明,为了本小节的目的,结构类型和联合类型对象的未命名成员不参与初始化。结构对象的未命名成员即使在初始化之后也具有不确定的值。

然而,在GNU GCC实现中,省略的成员被初始化为零或类似零的适合类型的值。如GNU GCC文档第6.27节中所述:

省略的字段成员隐式初始化,与具有静态存储持续时间的对象相同。

根据官方博客文章c++一致性路线图,微软Visual c++编译器应该从2013版开始支持指定的初始化器。MSDN Visual Studio文档中的初始化联盟和Initializers的文章建议将未命名成员初始化为类似于GNU GCC的类似于零的适当值。

ISO/IEC 9899:2011标准(通常称为C11)已经取代了ISO/IEC 9899:1999,在第6.7.9节初始化中保留了指定的初始化器。决议草案还保留第9段不变。

新的ISO/IEC 9899:2018标准(通常称为C18)已经取代了ISO/IEC 9899:2011,在第6.7.9节初始化中保留了指定的初始化器。决议草案还保留第9段不变。

其他回答

a = (MYTYPE){true, 15, 0.123};

会在C99做得很好吗

如果MS没有更新到C99, MY_TYPE a = {true,15,0.123};

我一直在寻找一个很好的方法来初始化我的结构,我已经使用下面的(C99)。这让我可以以与普通类型相同的方式初始化单个结构或结构数组。

typedef struct {
    char *str;
    size_t len;
    jsmntok_t *tok;
    int tsz;
} jsmn_ts;

#define jsmn_ts_default (jsmn_ts){NULL, 0, NULL, 0}

这可以在代码中使用:

jsmn_ts mydata = jsmn_ts_default; /* initialization of a single struct */

jsmn_ts myarray[10] = {jsmn_ts_default, jsmn_ts_default}; /* initialization of
                                                    first 2 structs in the array */

你差不多懂了……

MY_TYPE a = { true, 15, 0.123 };

快速搜索'struct initialize c'显示了这一点

添加到所有这些好的回答总结如何在C中初始化一个结构(联合和数组),特别是设计的初始化器。

标准的初始化

struct point 
{
    double x;
    double y;
    double z;
}

p = {1.2, 1.3}; 

指定初始化

指定初始化式出现于ISO C99,是在C语言中初始化struct、union或数组时的一种不同的、更动态的初始化方式。

与标准初始化的最大区别是,您不必按固定顺序声明元素,也可以省略元素。

摘自GNU指南:

标准C90要求初始化式的元素以固定的顺序出现,与初始化数组或结构中元素的顺序相同。

在ISO C99中,您可以以随机顺序给出元素,指定它们应用的数组下标或结构字段名,GNU C也允许在C90模式中作为扩展


例子

1. 数组索引

标准的初始化

int a[6] = { 0, 0, 15, 0, 29, 0 };

指定的初始化

int a[6] = {[4] = 29, [2] = 15 }; // or
int a[6] = {[4]29 , [2]15 }; // or
int widths[] = { [0 ... 9] = 1, [10 ... 99] = 2, [100] = 3 };

2. 结构或联合:

标准的初始化

struct point { int x, y; };

指定的初始化

struct point p = { .y = 2, .x = 3 }; or
struct point p = { y: 2, x: 3 };

3.将命名元素与连续元素的普通C初始化结合起来:

标准的初始化

int a[6] = { 0, v1, v2, 0, v4, 0 };

指定的初始化

int a[6] = { [1] = v1, v2, [4] = v4 };

4. 其他:

标记数组初始化式的元素

int whitespace[256] = { [' '] = 1, ['\t'] = 1, ['\h'] = 1,
                        ['\f'] = 1, ['\n'] = 1, ['\r'] = 1 };

写一系列的“。”' = '前面的Fieldname '和' [index] '指示符指定要初始化的嵌套子对象

struct point ptarray[10] = { [2].y = yv2, [2].x = xv2, [0].x = xv0 };

指南

Designated-initializers-c | geeksforgeeks.org using-designated-initializers Tutorialspoint.com |designing-initializs-inc