C和指针 第十章 结构和联合 (一)

结构体:

聚合数据类型是指,能够同时存储超过一个的单独数据,C语言中有两个聚合数据类型,数组和结构体。数组中储存的类型必须相同,元素通过下标和指针引用来访问的。

结构体也是一些值的集合,但是结构体中每个元素的值的类型可以不同,每个元素都有自己的名字,和整数字符一样,结构体也是标量类型

结构体声明:

struct  tag  {  memeber-list   }  variable-list;

结构体声明要包含所有的成员的类型和名字

struct  tag {
  char *name;
  int    *age;
} people;

其中,tag时标签字段,可以在后续的声明中使用tag代表成员列表:

struct tag animal;

此处的声明和上面的people一样,animal和people对于编译器来说,是两个不同的类型的。

如果需要定义相同的类型,可以通过typedef来创建新类型:

typedef struct  {
  char *name;
  int    *age;
} animal;

现在animal是一新的结构体类型,可以直接使用他进行赋值

animal dog;

结构体的成员:

结构体的成员可以时标量,指针,和其他结构体等等,但是不可以是自身类型的成员,即自己。结构体成员可以通过点号访问

#include <stdio.h>

typedef struct {
    char *name;
    int   age;
} animal;

int main()
{
    animal dog = { "ted", 3};
    printf("%s", dog.name);

    return 0;
}

运行结果:

 

如果是指向结构体的指针,可以先解引用操作,然后在使用点操作符,C有个快捷的方法,使用 -> 访问结构体指针指向的结构体成员,上面代码可以如下改写:

animal dog = { "ted", 3};
animal *ptr = &dog;
printf("%s", ptr -> name);

结构体时不可以包含自身类型的成员的,不过可以包含指向自己类型的指针的。

#include <stdio.h>
struct SELF {
    char *name;
   //selfPtr是指向自身类型的指针 struct SELF * selfPtr; } animal; int main() { struct SELF dog = { "ted", NULL}; struct SELF cat = {"mimi", &dog}; printf("%s", cat.selfPtr -> name); return 0; }

运行结果:

注意:下面的这种声明是非法的,因为类型名直到声明结束才定义

typedef struct {
    char *name;
    struct SELF * selfPtr;
} SELF;

解决方法是添加一个结构标签。

typedef struct SELF {
    char *name;
    struct SELF * selfPtr;
} animal;

如果两个结构体相互依赖引用,一个结构体包含另一个结构体,另一个结构体包含这个结构体。那么可以使用不完整声明,先声明一个结构体作为标识符,然后使用他,然后把成员和标识符关联。

  //声明表示符,在A中使用
    struct B;
    struct A {
        struct B *bPtr;
    };

    //关联成员
    struct B {
        struct A *aPtr;
    };

结构体的初始化,类似数组,大括号内部包含逗号分割的初始值,如果初始值列表不够用,剩余的结构成员将使用缺省值初始化。

    struct {
        char *name;
        int age;
    } me = {"yangxunwu", 24};

结构体的储存分配:

由于结构体类别的成员的类型可以不一样,编译器在给成员列表分配内存时,一个接着一个的分配,只有当储存成员需要满足边界对齐时,成员之间会出现储存间隙。系统禁止在一个结构的起始位置跳过几个字节来满足边界对齐要求,因此所有的结构的起始储存位置,必须是边界要求最严格的数据类型所要求的位置,如下:

#include <stdio.h>

int main()
{
    struct {
        char ch1;
        int age;
        char ch2;
    } stc1;

    struct {
        int age;
     char ch1;
     char ch2; } stc2; printf("%lu %lu", sizeof(stc1), sizeof(stc2)); return 0; }

可以通过重新排序成员,让边界对齐要求高的先出现,如上运行:

int类型占四个字节,char占一个字节,因为要从起始位置对齐,所以stc1的第一个char占4字节,而stc2中,age占四个字节,ch1和ch2在一起占四个字节。sizeof可以返回一个long unsigned int表示整个结构体长度。

stddef.h宏 offsetof(type, member) 返回指定成员,距离结构开始偏移几个字节。

 

#include <stdio.h>
#include <stddef.h>

int main()
{
    struct stc1 {
        char ch1;
        int age;
        char ch2;
    } ;

    struct stc2 {
        int age;
        char ch1;
        char ch2;
    } ;

    printf("%lu %lu", offsetof(struct stc1, ch2), offsetof(struct stc2, ch2));

    return 0;
}

 运行:

原文地址:https://www.cnblogs.com/yangxunwu1992/p/5831182.html