- 写一个宏,计算结构体中某变量相对于首地址的偏移,并给出说明
- 判断大小端
代码1:
//写一个宏,计算结构体中某变量相对于首地址的偏移,并给出说明
//定义的这个宏将结构体里边的变量的地址取出后再强转成char型,然后进行相减。
//注意:&s == &s.num (进行验证:请看代码2)
#define OFFSET(a, b) ((char*)(&b) - (char*)(&a))
#include <stdio.h>
typedef struct S{
double num ;
char ch ;
}S;
int main()
{
S s ;
printf("此结构体第二个变量相对于首地址的偏移为: %d\n", OFFSET(s, s.ch)) ;
return 0 ;
}
运行结果:
代码2:测试程序
//测试程序:验证&member 是否与 &member.sch_num 相等
#include <stdio.h>
typedef struct Node{
int sch_num ;
char add[20] ;
}Node ;
int main()
{
Node member = {10101, "beijing"} ; //定义结构体成员
printf("结构体地址为: %p\n", &member) ;
printf("结构体成员第一个变量首地址为: %p\n", &member.sch_num) ;
return 0 ;
}
运行结果:
代码3.1:联合体法判断大小端
//联合体判断大小端
#include <stdio.h>
union Un
{
int a ;
char i ;
};
int main()
{
union Un u ;
u.a = 0x12345678 ;
//在int的四个字节里,i占一个字节。通过打印i空间里的值,看看i所占的空间里值是多少
//若值为12,则表示大端存储;若为78,则表示小端存储。
printf("联合体判断大小端:\n") ;
printf("u.i = %x\n", u.i) ;
return 0 ;
}
运行结果:
代码3.2:指针法判断大小端
//指针法判断大小端
#include <stdio.h>
int main()
{
int a = 0x12345678 ;
char* p = (char*)&a ;
//定义一个char类型的指针,将a进行&a并进行强转,打印出p所指的内存里的值
//若值为12,则表示大端存储;若为78,则表示小端存储。
printf("指针法判断大小端:\n") ;
printf("%x\n", *p) ;
return 0 ;
}
运行结果:
附加:
目前Intel的80x86系列芯片是唯一还在坚持使用小端的芯片,ARM芯片默认采用小端,但可以切换为大端;而MIPS等芯片要么采用全部大端的方式储存,要么提供选项支持大端——可以在大小端之间切换。另外,对于大小端的处理也和编译器的实现有关,在C语言中,默认是小端(但在一些对于单片机的实现中却是基于大端,比如Keil 51C),Java是平台无关的,默认是大端。在网络上传输数据普遍采用的都是大端。(摘自百度百科)