当前位置:首页 > 单片机 > C语言编程
[导读]01.宏定义用do{}while(0)如果定义的宏函数后面有多条语句,使用这样的方式会有问题: #defineFUNC()func1();func2() if(bRunF)   FUNC();展开宏定义后会变成: if(bRunF)   func1();   func2();逻辑...

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);除了这个,还有

 #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. 用异或运算实现数据交换
交换俩变量数据,一般做法是:

 // 方法1
 temp = 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;
14. 判断一个整数是否为2的幂
也许你会不断地将这个数除以2,除到底,然而Linux kernel有个巧妙的办法:

 #define is_power_of_2(n) ((n) != 0
本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除( 邮箱:macysun@21ic.com )。
换一批
延伸阅读
关闭