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);

除了这个,还有

#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)]))

未完待续……

我打算写100条,你还不关注我?
在这里插入图片描述

你可能感兴趣的:(C语言)