當咱們新建一個工程的時候,在Supporting FIles文件下會看到一個以 -Prefix.pch結尾文件的文件,pch全稱是「precompiled header」,也就是預編譯頭文件,該文件裏存放的工程中一些不常被修改的代碼,好比經常使用的框架頭文件,這樣作的目的提升編譯器編譯速度。咱們知道當咱們修改一個工程中某個文件代碼時候,編譯器並非從新編譯全部全部文件,而是編譯改動過文件的,假如pch中某個文件修改了,那麼pch整個文件裏包含的的其餘文件也會從新編譯一次,這樣就會消耗大量時間,因此它裏面添加的文件最好是是不多變更或不變更的頭文件或者是預編譯的代碼片斷; html
在新建一個工程時,pch後綴文件裏代碼是 小程序
- #import <Availability.h>
-
- #ifndef __IPHONE_4_0
- #warning "This project uses features only available in iOS SDK 4.0 and later."
- #endif
-
- #ifdef __OBJC__
- #import <UIKit/UIKit.h>
- #import <Foundation/Foundation.h>
- #endif
或許你會以爲這預編譯代碼不多,可是你能夠查看一下UIKit.h的定義文件中
- //
- // UIKit.h
- // UIKit
- //
- // Copyright (c) 2005-2011, Apple Inc. All rights reserved.
- //
-
- #import <UIKit/UIKitDefines.h>
- #import <UIKit/UIAccelerometer.h>
- #import <UIKit/UIAccessibility.h>
- #import <UIKit/UIActivityIndicatorView.h>
- #import <UIKit/UIAlert.h>
- #import <UIKit/UIApplication.h>
- #import <UIKit/UIBarButtonItem.h>
- #import <UIKit/UIBarItem.h>
- #import <UIKit/UIBezierPath.h>
- #import <UIKit/UIButton.h>
- #import <UIKit/UIColor.h>
- #import <UIKit/UIControl.h>
- #import <UIKit/UIDataDetectors.h>
- #import <UIKit/UIDatePicker.h>
- #import <UIKit/UIDevice.h>
- #import <UIKit/UIDocument.h>
- #import <UIKit/UIDocumentInteractionController.h>
- #import <UIKit/UIEvent.h>
- #import <UIKit/UIFont.h>
- #import <UIKit/UIGeometry.h>
- #import <UIKit/UIGestureRecognizer.h>
- #import <UIKit/UIGraphics.h>
- #import <UIKit/UIImage.h>
- #import <UIKit/UIImagePickerController.h>
- #import <UIKit/UIImageView.h>
- #import <UIKit/UIInterface.h>
- #import <UIKit/UILabel.h>
- #import <UIKit/UILocalNotification.h>
- #import <UIKit/UILocalizedIndexedCollation.h>
- #import <UIKit/UILongPressGestureRecognizer.h>
- #import <UIKit/UIManagedDocument.h>
- #import <UIKit/UIMenuController.h>
- #import <UIKit/UINavigationBar.h>
- #import <UIKit/UINavigationController.h>
- #import <UIKit/UINib.h>
- #import <UIKit/UINibDeclarations.h>
- #import <UIKit/UINibLoading.h>
- #import <UIKit/UIPageControl.h>
- #import <UIKit/UIPageViewController.h>
- #import <UIKit/UIPanGestureRecognizer.h>
- #import <UIKit/UIPasteboard.h>
- #import <UIKit/UIPickerView.h>
- #import <UIKit/UIPinchGestureRecognizer.h>
- #import <UIKit/UIPopoverController.h>
- #import <UIKit/UIPopoverBackgroundView.h>
- #import <UIKit/UIPrintError.h>
- #import <UIKit/UIPrintFormatter.h>
- #import <UIKit/UIPrintInfo.h>
- #import <UIKit/UIPrintInteractionController.h>
- #import <UIKit/UIPrintPageRenderer.h>
- #import <UIKit/UIPrintPaper.h>
- #import <UIKit/UIProgressView.h>
- #import <UIKit/UIReferenceLibraryViewController.h>
- #import <UIKit/UIResponder.h>
- #import <UIKit/UIRotationGestureRecognizer.h>
- #import <UIKit/UIScreen.h>
- #import <UIKit/UIScreenMode.h>
- #import <UIKit/UIScrollView.h>
- #import <UIKit/UISearchBar.h>
- #import <UIKit/UISearchDisplayController.h>
- #import <UIKit/UISegmentedControl.h>
- #import <UIKit/UISlider.h>
- #import <UIKit/UISplitViewController.h>
- #import <UIKit/UIStepper.h>
- #import <UIKit/UIStoryboard.h>
- #import <UIKit/UIStoryboardPopoverSegue.h>
- #import <UIKit/UIStoryboardSegue.h>
- #import <UIKit/UIStringDrawing.h>
- #import <UIKit/UISwipeGestureRecognizer.h>
- #import <UIKit/UISwitch.h>
- #import <UIKit/UITabBar.h>
- #import <UIKit/UITabBarController.h>
- #import <UIKit/UITabBarItem.h>
- #import <UIKit/UITableView.h>
- #import <UIKit/UITableViewCell.h>
- #import <UIKit/UITableViewController.h>
- #import <UIKit/UITapGestureRecognizer.h>
- #import <UIKit/UITextField.h>
- #import <UIKit/UITextInput.h>
- #import <UIKit/UITextInputTraits.h>
- #import <UIKit/UITextView.h>
- #import <UIKit/UIToolbar.h>
- #import <UIKit/UITouch.h>
- #import <UIKit/UIVideoEditorController.h>
- #import <UIKit/UIView.h>
- #import <UIKit/UIViewController.h>
- #import <UIKit/UIWebView.h>
- #import <UIKit/UIWindow.h>
這些很多了吧,工程每次運行都編譯是否是很費時間,這些是蘋果公司內部定義的標準頭文件,咱們不能也沒有權限修改這些頭文件定義內容,因此,當放到pch文件中會加速編譯過程;
再來看看咱們開源中國iOS客戶端pch文件 api
咱們看到有這樣些文件也被添加到裏面,可能會想難道這些頭文件變化不大嗎? 數組
- //添加的預編譯
- #import "ASIHTTPRequest.h"
- #import "ASIFormDataRequest.h"
- #import "ASIHTTPRequestDelegate.h"
- #import "ASIHTTPRequestConfig.h"
- #import "TBXML.h"
- #import "TBXML+HTTP.h"
- #import "TBXML+Compression.h"
- #import "Config.h"
- #import "EGORefreshTableHeaderView.h"
- #import "DataSingleton.h"
- #import "ImgRecord.h"
- #import "IconDownloader.h"
- #import "MBProgressHUD.h"
- #import "GCDiscreetNotificationView.h"
- #import "NdUncaughtExceptionHandler.h"
- #import "JSNotifier.h"
- #import "AFOSCClient.h"
- #import "AFHTTPRequestOperation.h"
- #import "AFXMLRequestOperation.h"
其實,這些文件特殊之處在於他們都是第三方類庫的頭文件,第三方類庫將一些對象進行高度封裝,留下接口,而後咱們根據類庫接口直接調用就能夠,這些第三方類庫通常都比iOS原生自帶的更加簡單易用,好比TBXML解析庫,比iOS自帶的NSXMLPaser解析器速度功能上都會好一些;
還有一些宏定義都是比較經常使用方式的宏定義,好比定義的開源中國社區的api接口,這些接口變得固然不多了; 網絡
而後就剩下最後面的 app
- #ifdef DEBUG
- #define debugLog(...) NSLog(__VA_ARGS__)
- #define debugMethod() NSLog(@"%s", __func__)
- #else
- #define debugLog(...)
- #define debugMethod()
- #endif
工程有Debug Version和Release Version,Debug Version是程序開發過程當中版本,它包含了全部調試信息,一些經常使用的NSLog打印日誌,在程序調試過程工根據咱們設置的調試信息能夠看出什麼地方出錯,咱們在運行運行一個小程序的時候,會不會首先就想到進行斷點調試呢,應該是首先想着NSLog一下,看看哪一個函數方法沒執行,看看是否是哪一個數組的值沒取出來。Release Version是發佈版本,不打印NSLog能夠加快程序運行速度,減小內存使用。 可是到一個大工程中,會有不少不少這樣的NSLog,在咱們工程完美運行的時候,發佈Release 版本的時候,難道咱們去一行行的註釋調NSLog嗎?假如工程如今原來基礎上發佈一個version 1.2版本的,咱們在修改程序的時候豈不是還把原來註釋給取消,那就很麻煩很麻煩了。
因此,此處用到了宏指令 框架
上段代碼的意思就是 用宏指令作一個判斷,若是DEBUG爲真,則編譯#ifdef到#endif宏定義,不然編譯器就不編譯; ide
這個DEBUG在哪設置呢, 函數
在 "Target > Build Settings > Preprocessor Macros > Debug" 裏有一個"DEBUG=1"。
![](http://static.javashuo.com/static/loading.gif)
post
如今咱們來作一個測試:
取一個宏指令放到OSAppDelegate.m的application:didFinishLaunchingWithOptions:方法中,並用同一個NSLog作一個對比;
NSLog(@"%s", __func__);
debugMethod();
首先設置爲Debug模式下,Product-->Edit Scheme
![](http://static.javashuo.com/static/loading.gif)
跳轉到這個界面
![](http://static.javashuo.com/static/loading.gif)
當我設置Build Configuration成Debug時,打印效果圖
當我設置Build Configuration成Release的,打印時效果圖
當Run Test Profile Analyze Archive的時候,均可以根據須要設置Debug和Release兩個模式運行;
因此咱們徹底能夠用一個宏指令來設置是否打印調試信息;