牛客网刷题:
-
数组与指针的理
// 在 gcc 环境下,已知数组 int arr[5]={1,2,3,4,5}; 则 *(*(&arr+1)-1) 的结果为?
/*
解答:在这里我们要区分一下&arr与arr的区别:
arr通常用来代表int地址中的开始地址,而&arr是一个指向数组的指针,类型为'int (*) [5]'
*/
-
类型字节大小以及计算机存储数据形式
short int i = 32769;
printf("%d\n",i);
//求输出结果
// 在计算机中使用补码,显示使用的源码
// 题中32769的源码为0000 0000 0000 0000 1000 0000 0000 0001,正数源码等于反码等于补码,但i的类型为short int 在内存中只占两个字节所以存储的源码值为1000 0000 0000 0001,当转化为源码输出时为1111 1111 1111 1111(其中第一位代表符号)。转化为十进制为-32767。
-
二义性的宏定义
/*
二义性的宏定义:
定义:在宏定义中存在多种解释或多种可能的解释
原理:宏定义是将一个标识符直接替换为特定的文本,不进行一些逻辑上的考虑
*/
#define SQUARE(x) x * x
int result = SQUARE(2 + 3);
/*
上述代码被展开后将会是:'2 + 3 * 2 + 3',而不是期望的'(2 + 3) * (2 + 3)'
这就是宏定义中会出现的二义性现象
*/
// 例题
#define add(a,b) a+b
int main()
{
printf("%d\n",5*add(3,4));
return 0;
}// 输出结果为19,并不是35
结构体的字节对齐的理解:
// 今天在刷题时遇到一个内存分配的问题,我们先看题目
union package {
char head;
int body;
};
struct message {
char id;
int crc;
union package pack;
};
int main() {
printf("size=%d\n",sizeof(struct message));
return 0;
}
/*
我的思路:
由上述代码可以看出,他给我们定义个一个结构体和一个联合体,联合体的内存通常为其成员最大成员值的大小,所以package占用了4个字节。而结构体内char占一个字节,int占4个字节,package占4个字节,所以整体为9个字节。(但是答案为12个字节)
答案解析:
在结构体中会对成员内存进行扩充,此题中char类型被扩充到4个字节。所以结果为12个字节。
*/