C语言的奇技淫巧之一
▍01. 宏定义用do{}while(0)
如果定义的宏函数后面有多条语句,使用这样的方式会有问题:
#define FUNC() func1(); func2()
if(bRunF)
FUNC();
展开宏定义后会变成:
if(bRunF)
func1();
func2();
逻辑就不对了。可以用这一的方式解决,非常好用:
#define FUNC() do{func1(); func2();}while(0)
▍02. 数组的初始化
假如给arr的第2~6元素初始化为5,也许你会
int arr[10] = {0, 5, 5, 5, 5, 5, 0, 0, 0, 0};
现在告诉你C99可以这样:
int arr[10] = {[1 ... 5] = 5};
▍03. 数组的访问
你想取数组的第6个元素(下标为5),教科书教你这样做:
int arr[10] = {0, 1, 2, 3, 4, 5, 6, 7, 8, 9};
int n1 = arr[5];
int n2 = *(arr+5);
其实你可以:
int arr[10] = {0, 1, 2, 3, 4, 5, 6, 7, 8, 9};
int n = 5[arr];
也不会有错,实际上arr[5]对应*(arr+5)
,而5[arr]对应*(5+arr)
,没多大区别。
▍04. 结构体的初始化
结构体的初始化,传统的做法是:
typedef struct
{
int a;
int x;
int y;
int z;
char b;
short c;
}S;
S s = {100, 0, 0, 0, 'A', 0x12);
对于C99,其实你可以:
typedef struct
{
int a;
int x;
int y;
int z;
char b;
short c;
}S;
S s = {
.a = 100,
.b = 'A',
.c = 0x12
};
▍05. 用include
的方式初始化大数组
double array[SIZE][SIZE] = {
#include "float_values.txt"
}
▍06. Debug时输出文件名、函数名、行号等
#define DEBUG_INFO() fprintf(stderr,"[DEBUG]%s:%d %s\n", __FILE__, __LINE__, __FUNCTION__);
▍07. C语言有-->
“趋向于...”操作符?
int main(void)
{
int n = 10;
while(n --> 0 ) // n goes to 0
{
printf("%d ", n);
}
printf("\n");
}
实际上C语言没有这个-->
操作符,是--
和>
的组合而已
while( n-- > 0 )
▍08. 获得任意类型数组的元素数目
#define NUM_OF(arr) (sizeof (arr) / sizeof (*arr))
▍09. 判断运行环境的大小端
Linux有以下代码:
static union {
char c[4];
unsigned long l;
} endian_test = { { 'l', '?', '?', 'b' } };
#define ENDIANNESS ((char)endian_test.l)
printf("ENDIANNESS: %c\n", ENDIANNESS);
▍10. 编译时做条件检查
Linux Kernel有以下代码
/* Force a compilation error if condition is true */
#define BUILD_BUG_ON(condition) ((void)sizeof(char[1 - 2*!!(condition)]))
例如,在某些平台为了防止内存对齐问题,检查一个结构体或者一个数组的大小是否为8的倍数。
BUILD_BUG_ON((sizeof(struct mystruct) % 8) != 0);
未完待续……
我打算写100条,你还不关注我?
少年,等等……点个“在看”再走,分享给更多人学习。