一、写一个宏,可以将一个整数的二进制位的奇数位和偶数位交换。
//写一个宏,可以将一个整数的二进制位的奇数位和偶数位交换。
#include<stdio.h>
#define SWAP(x) ((((x)&(0xAAAAAAAA))>>1)+(((x)&(0x55555555))<<1))
int main()
{
int ret = SWAP(5);
printf("%d\n", ret);
return 0;
}
这道题其实是先分别得到一个数的偶数位和奇数位,再将偶数位的所有数右移一位,奇数位的所有数左移一位,最后相加,得到目标数字。
0xaaaa aaaa 其实是二进制的1010 1010 1010 1010 1010 1010 1010 1010,可以得到偶数位;
0x5555 5555是二进制0101 0101 0101 0101 0101 0101 0101 0101,按位与后得到奇数位。
二、写一个宏,计算结构体中某变量相对于首地址的偏移
我们知道有一个宏可以计算结构体成员的偏移量:offsetof。
先来看看offsetof:
size_t offsetof( structName, memberName ); //offset返回值得单位是字节
我们知道,由于结构体的对齐方式,我们无法直接计算结构体成员的偏移量,而一个结构体成员的偏移量=该成员地址-结构体地址,那么当结构体地址是0,结构体成员的地址不就是该成员的偏移量吗?
所以就有:
#define OFFSETOF(STRUCTName,MEMBERName) (size_t)&((STRUCTName*)0)->MEMBERName
这其实就是把0强行转成结构体类型的指针,假设它就是当前结构体的指针,再看此时成员的地址。
//写一个宏,计算结构体中某变量相对于首地址的偏移,并给出说明
//考察:offsetof宏的实现
#include<stdio.h>
#define OFFSETOF(STRUCTName,MEMBERName) (size_t)&((STRUCTName*)0)->MEMBERName
struct S
{
int i;
char c;
double f;
};
int main()
{
size_t ret = OFFSETOF(struct S, f);
printf("%u\n", ret);
return 0;
}