C语言的实用妙招,总有一款适合你
扫描二维码
随时随地手机看文章
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);除了这个,还有 #define BUILD_BUG_ON_ZERO(e) (sizeof(struct{int : -!!(e);}))
#define BUILD_BUG_ON_NULL(e) ((void*)sizeof(struct{int : -!!(e);}))
#define BUILD_BUG_ON(condition) ((void)BUILD_BUG_ON_ZERO(condition))
#define MAYBE_BUILD_BUG_ON(condition) ((void)sizeof(char[1 - 2 * !!(condition)]))
11. 用异或运算实现数据交换
交换俩变量数据,一般做法是: // 方法1temp = a;
a = b;
b = temp;
// 方法2
a=a b;
b=a-b;
a=a-b;方法1需要第三个变量,方法二存在数据溢出可能,可以尝试下以下方法:a = a ^ b;
b = a ^ b;
a = a ^ b;
12. 判断语句中把const
数值放在前面
通常条件语句写成if(n == 0){ /*...*/ }但是,有可能手误写成if(n = 0){ /*...*/ }这种错误只有机器在运行时候知道,而人不一定能发现这种bug。把数值放在前面就不怕了,==
写成=
,编译器就知道if(0 == n){ /*...*/ }13. 用冒号表达式替代if...else...
语句
这个用法应该很普遍了,不算什么特别的技巧了。 if(y < 0){
x = 10;
}
else
{
x = 20;
}可以改成以下一行代码即可 x = (y < 0) ? 10 : 20;