C 语言中同意将值为 0 的变量强制转换成任一类型的指针,转换结果是一个NULL指针。

(type*)0 // 一个 type 类型的NULL指针

用这个指针訪问结构体内的成员是非法的,可是

&(((type*)0)->field)

是为了计算 field 的地址 ,编译器不会产生訪问 field 的代码。仅仅会依据 type 的布局和起始地址在编译期计算这个地址(常量)。而又由于初始地址为 0,故该地址的值就是该结构体成员相对于结构体基址的偏移。

(size_t)&(((type*)0)->field)

结构体的 sizeof 的相关知识:

对 32 位操作系统。默认按 4 字节对齐;

sizeof(union),结果是 union 占用内存最大成员的大小(各成员共享内存)

编译器是依照什么样的原则进行对齐的?

  • 1.数据类型自身的对齐值:

    对于char型数据,其自身对齐值为1,对于short型为2,对于int, float类型,其自身对齐值为4,对于double类型其自身对齐值为8,单位字节。

  • 2.结构体或者类的自身对齐值:其成员中自身对齐值最大的那个值。

  • 3.指定对齐值:#pragma pack (value)时的指定对齐值value。

  • 4.数据成员、结构体和类的有效对齐值:自身对齐值和指定对齐值中小的那个值。

  • 5.对齐后的长度必须是成员中最大的对齐參数的整数倍
#include <iostream>
using namespace std;
#define ISUNSIGNED(x) ((x > 0) && (~x > 0))
#define OffSet(type, field) ((size_t)&(((type*)0)->field))
int main()
{
struct A{
char a;
int b;
float c;
double d;
char e;
union B{
char f[100];
struct C{
int g[50];
} h;
}i;
bool j;
} s;
cout << sizeof(A) << endl; // 232
cout << "a 的起始偏移:"<< OffSet(A, a) << endl; // 0
cout << "b 的起始偏移:"<< OffSet(A, b) << endl; // 4
cout << "c 的起始偏移:"<< OffSet(A, c) << endl; // 8
cout << "d 的起始偏移:"<< OffSet(A, d) << endl; // 16
cout << "e 的起始偏移:"<< OffSet(A, e) << endl; // 24
cout << "i 的起始偏移:"<< OffSet(A, i) << endl; // 28
cout << "j 的起始偏移:"<< OffSet(A, j) << endl; // 228 int p = 1;
unsigned int q = 1;
cout << p << " " << ~p << endl; // 1 -2
cout << q << " " << ~q << endl; // 1 4294967294
cout << "p是否是无符号数:" << ISUNSIGNED(p) << endl; // 0
cout << "q是否是无符号数:" << ISUNSIGNED(q) << endl; // 1
return 0;
}

以上结构体分析:

struct A{
char a;
int b;
float c;
double d;
char e;
union B{
char f[100];
struct C{
int g[50];
} h;
}i;
bool j;
} s;
  • 结构体 A 的自身对齐值为最大的成员的自身对齐值,为 8 (double d), 注意内部结构体 C 的最大对齐值是 4 (int)。Union B 最大对齐值是 4 (int);
  • 成员char a 的本身对齐值是 1,结构体对齐值是 8。按 1 字节对齐;
  • 成员 int b 的 本身对齐值是 4。结构体对齐值是 8,按 4 字节对齐,它的起始偏移地址为 4。
  • 成员 float c 的 本身对齐值是 4,结构体对齐值是 8,按 4 字节对齐。它的起始偏移地址为 8。
  • 成员 double d 的 本身对齐值是 8,结构体对齐值是 8。按 8 字节对齐,它的起始偏移地址为 16;
  • 成员 char e 的 本身对齐值是 1,结构体对齐值是 8,按 1 字节对齐。它的起始偏移地址为 24;(16+8)
  • 成员 Union B i 的 本身对齐值是 4,结构体对齐值是 8,按 4 字节对齐。它的起始偏移地址为 28;
  • 成员 bool j 的 本身对齐值是 1,结构体对齐值是 8。按 1 字节对齐。它的起始偏移地址为 228 ;(28+200)
  • 结构体对齐值是 8, 因此结构体 A 的大小为 232 (228+1 补齐成 8 的倍数)

一个成员在内存中补位所占的内存,决定于该成员之后的成员类型。

參考:《程序猿笔试面试宝典》

怎样求结构体成员的偏移地址 || 结构体的 sizeof 总结的更多相关文章

  1. 套接字编程简介: IPV4套接字地址结构/ 通用套接字地址结构/ IPV6套接字地址结构/新通用套接字地址结构

    IPv4套接字地址结构通常也称为“网际套接字地址结构”,它以sockaddr_in命名,定义在<netinet/in.h>头文件中. struct in_addr { in_addr_t ...

  2. C语言根据结构体成员变量的地址,得到结构体的地址

    看nginx代码时发现双链表使用的是这种方法,记录一下 给出一个实例来说明 struct father_t {    int a;    char *b;    double c;}f;char *p ...

  3. 指针直接赋值为整型AND利用宏定义求结构体成员偏移量

    首先我们要更正一个很熟悉的概念,那就是指针不仅仅是“地址”,指针还有一个很重要的特性,那就是“类型”. 指针初始化时,“=”的右操作数; 除外,该语句表示指针为空): 所以 ; 这样的代码是不允许的. ...

  4. 由结构体成员地址计算结构体地址——list_entry()原理详解

    #define list_entry(ptr, type, member) container_of(ptr, type, member) 在进行编程的时候,我们经常在知道结构体地址的情况下,寻找其中 ...

  5. Linux中的两个经典宏定义:获取结构体成员地址,根据成员地址获得结构体地址;Linux中双向链表的经典实现。

    倘若你查看过Linux Kernel的源码,那么你对 offsetof 和 container_of 这两个宏应该不陌生.这两个宏最初是极客写出的,后来在Linux内核中被推广使用. 1. offse ...

  6. 【C语言】这种求结构体成员大小的方法,你可能需要了解一下~

    在C语言编程中,有时候需要知道某结构体中某成员的大小,比如使用堆内存来存储结构体中的某成员时,需要知道该成员的大小,才好确定所需申请的空间大小.求某结构体中某成员的大小,你会怎么做? 例子: type ...

  7. C/C++ 结构体成员在内存中的对齐规则(转载)

    这几天在看王艳平的<windows 程序设计>,第5章讲解了MFC框架是怎么管理窗口句柄到窗口实例之间的映射,用到了两个类CPlex和CMapPtrToPtr,用于管理内存分配的类(避免因 ...

  8. C/C++ 结构体成员在内存中的对齐规则

    这几天在看王艳平的<windows 程序设计>,第5章讲解了MFC框架是怎么管理窗口句柄到窗口实例之间的映射,用到了两个类CPlex和CMapPtrToPtr,用于管理内存分配的类(避免因 ...

  9. 【PE结构】由浅入深PE基础学习-菜鸟手动查询导出表、相对虚拟地址(RVA)与文件偏移地址转换(FOA)

    0 前言 此篇文章想写如何通过工具手查导出表.PE文件代码编程过程中的原理.文笔不是很好,内容也是查阅了很多的资料后整合出来的.希望借此加深对PE文件格式的理解,也希望可以对看雪论坛有所贡献.因为了解 ...

随机推荐

  1. Wannafly挑战赛23 A 字符串

    题目描述 小N现在有一个字符串S.他把这这个字符串的所有子串都挑了出来.一个S的子串T是合法的,当且仅当T中包含了所有的小写字母.小N希望知道所有的合法的S的子串中,长度最短是多少. 输入描述: 一行 ...

  2. ARM-Linux基本开发步骤

    拿到一块YC2440(s3c2440)的开发板,经过几天的学习,我对arm-linux系统开发步骤有了一些认识.就以开发这个开发板为例,arm-linux开发工作大概分4个部分 1.       硬件 ...

  3. java null 空指针

    对于Java程序员来说,null是令人头痛的东西.时常会受到空指针异常(NPE)的骚扰.连Java的发明者都承认这是他的一项巨大失误.Java为什么要保留null呢?null出现有一段时间了,并且我认 ...

  4. xshell连接linux

    一些命令和快捷键: Ctrl + Alt 切换linux和windows的鼠标 Ctrl + c 或 Ctrl + d退出>状态 在xshell终端输入exit,退出与linux服务器的连接 登 ...

  5. Android开发调试无法连接到夜神模拟器的解决方法

    Android开发调试无法连接到夜神模拟器的解决方法: 一般原因是adb的版本不一致造成的!!!!!换成一样的就可以了. 在网上看到的方法,特记录下来: 1.任务管理器里看下,adb.exe以及nox ...

  6. 面试准备——redis

    https://blog.csdn.net/yangzhong0808/article/details/81196472 http://www.imooc.com/article/36399 http ...

  7. BZOJ 2005 [Noi2010]能量采集 ——Dirichlet积

    [题目分析] 卷积一卷. 然后分块去一段一段的求. O(n)即可. [代码] #include <cstdio> #include <cstring> #include < ...

  8. TeraTerm下载

    参考:http://blog.csdn.net/gxp/article/details/9169319 官方主页: http://ttssh2.sourceforge.jp/ 官方下载: http:/ ...

  9. P1168 中位数 (优先队列,巧解)

    题目描述 给出一个长度为N的非负整数序列A[i],对于所有1 ≤ k ≤ (N + 1) / 2,输出A[1], A[3], …, A[2k - 1]的中位数.即前1,3,5,……个数的中位数. 输入 ...

  10. HP-Unix安装Memcache问题

    安装环境 HP-UX 125RITEC B.11.31 U ia64 0942432495 gcc (GCC) 4.3.1 libevent-1.4.14b-stable.tar.gz memcach ...