iOS開發系列--C語言之預處理

概述

你們都知道一個C程序的運行包括編譯和連接兩個階段,其實在編譯以前預處理器首先要進行預處理操做,將處理完產生的一個新的源文件進行編譯。因爲預處理指令是在編譯以前就進行了,所以不少時候它要比在程序運行時進行操做效率高。在C語言中包括三類預處理指令,今天將一一介紹:函數

  1. 宏定義
  2. 條件編譯
  3. 文件包含

宏定義

對於程序中常常用到的一些常量或者簡短的函數咱們一般使用宏定義來處理,這樣作的好處是對於程序中全部的配置咱們能夠統一在宏定義中進行管理,並且因爲宏定義是在程序編譯以前進行替換相比定義成全局變量或函數效率更高。性能

//
//  main.c
//  Pretreatment
//
//  Created by Kenshin Cui on 14-6-28.
//  Copyright (c) 2014年 Kenshin Cui. All rights reserved.
//

#include <stdio.h>
#define PI 3.14 //宏定義通常大寫
#define R 10
#define S 2*PI*R //在另外一個宏裏面引用了上面的宏

int main(int argc, const char * argv[]) {
    float r=10.5;
    double area=PI*r*r;
    printf("area=%.2f\n",area);
    
    double a=S;
    printf("a=%.2f\n",a);
    printf("PI=3.14\n");//注意輸出結果不是3.14=3.14而是PI=3.14,字符串中的PI並不會被替換
#undef PI //強制終止宏定義,不然它的範圍一直到文件結束
    int PI=3.1415926;
    double area2=PI*r*r;
    printf("area2=%.2f\n",area2);
    
    
    return 0;
}

宏定義實際的操做就是在預處理時進行對應替換,這個階段無論語法是否正確,並且對於字符串中出現的宏名不會進行替換。宏定義的功能事實上是很是強大的,除了簡單的常量替換還能夠傳入參數:ui

//
//  1.2.c
//  Pretreatment
//
//  Created by Kenshin Cui on 14-7-17.
//  Copyright (c) 2014年 Kenshin Cui. All rights reserved.
//

#include <stdio.h>
#define SUM(a,b) a+b
#define SUB(a,b) (a-b)
#define MUL (a,b) (a*b) //這麼定義是錯誤的,預處理器會認爲宏名爲」MUL「,替換內容爲」(a,b) (a*b)「


int main(int argc, const char * argv[]) {
    
    int a=2,b=3,c,d;
    c=SUM(a, b);
    printf("c=%d\n",c); //結果:c=5
    d=SUM(a, b)*2;
    printf("d=%d\n"); //結果:8,爲何不是10呢?由於替換後:d=a+b*2也就是2+3*2=8
    
    int e=SUB(b, a)*2;
    printf("(b-a)*2=%d\n",e); //結果:2,若是SUB定義時不加括號這裏應該是-1
    
    return 0;
}
上面咱們能夠看出帶參數的宏功能很強大,有點相似於函數,同函數不一樣的是它只是簡單的替換,不涉及存儲空間分配,參數、返回值等問題,可是因爲它在預處理階段展開,因此通常效率較高。使用帶參數的宏須要注意的就是結果最好用括號括起來不然很容易出現問題(在上面的SUM例子中咱們應該已經看到了);還有一點就是帶參數的宏定義時名稱和參數之間不要有空格。

條件編譯

條件編譯其實就是在編譯以前預處理器根據預處理指令判斷對應的條件,若是條件知足就將對應的代碼編譯進去,不然代碼就根本不進入編譯環節(至關於根本就沒有這段代碼)。spa

//
//  main.c
//  Pretreatment
//
//  Created by Kenshin Cui on 14-06-28.
//  Copyright (c) 2014年 Kenshin Cui. All rights reserved.
//

#include <stdio.h>
#define COUNT 1

int main(int argc, const char * argv[]) {
    
//判斷是否認義了 COUNT 宏
#if defined(COUNT) //等價於:#ifdef COUNT,相反若是判斷沒有定義過則能夠經過#if !defined(COUNT)或者#ifndef COUNT
    printf("COUNT defined\n");
#endif
    
//判斷宏定義COUNT是否等於1
#if COUNT==1
    showMessage("hello,world!\n");
#else
    say();
#endif
    
    return 0;
}

文件包含

文件包含指令#include在前面也屢次使用過,這裏再次強調一下。首先使用#include「xxx」包含和使用#include <xxx>包含的不一樣之處就是使用<>包含時,預處理器會搜索C函數庫頭文件路徑下的文件,而使用「」包含時首先搜索程序所在目錄,其次搜索系統Path定義目錄,若是仍是找不到纔會搜索C函數庫頭文件所在目錄。code

另外在使用#include的時候咱們須要注意包含文件的時候是不能遞歸包含的,例如a.h文件包含b.h,而b.h就不能再包含a.h了;還有就是重複包含雖然是容許的(這裏指的是重複包含頭文件)可是這會下降編譯性能,不妨看一下下面的例子:blog

includeRepeat

上面有三段代碼,在main.c和person.h中都包含了message.h而main.c自身又包含了person.h,這樣程序在預處理階段會對包含內容進行替換,替換後mian.c中包含了兩個#include 「message.h」雖然沒有報錯,但這會影響編譯的性能,正確的作法應該是這樣的:遞歸

correctInclude

其實就是用宏定義判斷一個宏是否認義了,若是沒有定義則會定義這個宏,這樣以來若是已經包含過則這個宏定義確定已經定義過了,即便再包含也不會從新定義了,下面的代碼也就不會包含進去。字符串

相關文章
相關標籤/搜索