数据结构 / 结构体字节计算

1. 结构体的存储

  • 结构体各个成员的地址是连续的
  • 结构体变量的地址是第一个成员的地址

2. 64位操作系统8字节对齐

  • 结构体的总字节大小是各个成员字节的总和,字节的总和需要是最宽成员的倍数
  • 结构体的首地址是最宽成员的倍数
  • 结构体各个成员的偏移量是该成员字节的倍数,否则填充空字节

例子1:

cpp 复制代码
#include <string.h>
#include <stdlib.h>
#include <stdio.h>


struct A
{
    int a; //4 
    char b; //1 
    short c; //2 
    float d; //4 

}a;


int main(int argc, const char *argv[])
{
    printf("a=%p\n", &a);
    printf("a.a=%p\n", &a.a);
    printf("a.b=%p\n", &a.b);
    printf("a.c=%p\n", &a.c);
    printf("a.d=%p\n", &a.d);
    printf("sizieof(a)=%ld\n", sizeof(a));
    
    return 0;
}          

计算

运行结果

cpp 复制代码
a=0x55b0037eb018
a.a=0x55b0037eb018
a.b=0x55b0037eb01c
a.c=0x55b0037eb01e
a.d=0x55b0037eb020
sizieof(a)=12

例子2:

cpp 复制代码
#include <string.h>
#include <stdlib.h>
#include <stdio.h>


struct A
{
    int *a; //8 
    char b; //1 
    double c; //8 
    long d; //8 

} s1; 


int main(int argc, const char *argv[])
{
    printf("s1=%p\n", &s1);
    printf("s1.a=%p\n", &s1.a);
    printf("s1.b=%p\n", &s1.b);
    printf("s1.c=%p\n", &s1.c);
    printf("s1.d=%p\n", &s1.d);
    printf("sizieof(s1)=%ld\n", sizeof(s1));
    
    return 0;
}  

计算

运行结果

cpp 复制代码
s1=0x55e754e4e040
s1.a=0x55e754e4e040
s1.b=0x55e754e4e048
s1.c=0x55e754e4e050
s1.d=0x55e754e4e058
sizieof(s1)=32

例子3:

cpp 复制代码
#include <string.h>                                                                       
#include <stdlib.h>
#include <stdio.h>


struct S
{
    char a[5]; //
    int b; //

} s3; 


int main(int argc, const char *argv[])
{
    printf("s3=%p\n", &s3);
    printf("s3.a=%p\n", &s3.a);
    printf("s3.b=%p\n", &s3.b);
    printf("sizieof(s3)=%ld\n", sizeof(s3));
        
    return 0;
}

计算

运行结果

cpp 复制代码
s3=0x55da08b43018
s3.a=0x55da08b43018
s3.b=0x55da08b43020
sizieof(s3)=12

3. 32位操作系统,4字节对齐:

  • 结构体的总字节大小是各个成员字节的总和,字节的总和需要是最宽成员的倍数
    • 如果最宽成员是1,则是1的倍数
    • 如果最宽成员是2,则是2的倍数
    • 如果最宽成员是4,8,则是4的倍数
  • 结构体的首地址是最宽成员的倍数
  • 结构体各个成员的偏移量是该成员字节的倍数,否则填充空字节
    • 如果成员字节是1,则偏移量是1的倍数
    • 如果成员字节是2,则偏移量是2的倍数
    • 如果成员字节是4,8,则偏移量是4的倍数

例子4:

cpp 复制代码
#include <string.h>
#include <stdlib.h>
#include <stdio.h>


struct S5
{
    char a; //
    int *b; 
    double c;
    long d;
        

}s5 ;

struct S6
{
    char e[5];
    struct S5 s5; 
    int f;

}s6;


int main(int argc, const char *argv[])
{
    printf("s6.e=%p\n", &s6.e);
    printf("s6.s5.a=%p\n", &s6.s5.a);
    printf("s6.s5.b=%p\n", &s6.s5.b);                                                     
    printf("s6.s5.c=%p\n", &s6.s5.c);
    printf("s6.s5.d=%p\n", &s6.s5.d);
    printf("s6.f=%p\n", &s6.f);
    printf("sizieof(s6)=%ld\n", sizeof(s6));
        
    return 0;
}
~    

计算

运行结果

cpp 复制代码
s6.e=0x565cf040
s6.s5.a=0x565cf048
s6.s5.b=0x565cf04c
s6.s5.c=0x565cf050
s6.s5.d=0x565cf058
s6.f=0x565cf05c
sizieof(s6)=32
相关推荐
用户311879455921820 小时前
Kylin Linux 10 安装 glib2-devel-2.62.5-7.ky10.x86_64.rpm 方法(附安装包)
linux
涛啊涛21 小时前
Centos7非LVM根分区容量不足后扩容,对调硬盘挂载/
linux·磁盘管理
CYRUS_STUDIO2 天前
用 Frida 控制 Android 线程:kill 命令、挂起与恢复全解析
android·linux·逆向
熊猫李2 天前
rootfs-根文件系统详解
linux
dessler2 天前
Hadoop HDFS-高可用集群部署
linux·运维·hdfs
泽泽爱旅行2 天前
awk 语法解析-前端学习
linux·前端
轻松Ai享生活3 天前
5 节课深入学习Linux Cgroups
linux
Fanxt_Ja3 天前
【LeetCode】算法详解#15 ---环形链表II
数据结构·算法·leetcode·链表
christine-rr3 天前
linux常用命令(4)——压缩命令
linux·服务器·redis
三坛海会大神5553 天前
LVS与Keepalived详解(二)LVS负载均衡实现实操
linux·负载均衡·lvs