当咱们新建一个工程的时候,在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"。

post
如今咱们来作一个测试:
取一个宏指令放到OSAppDelegate.m的application:didFinishLaunchingWithOptions:方法中,并用同一个NSLog作一个对比;
NSLog(@"%s", __func__);
debugMethod();
首先设置为Debug模式下,Product-->Edit Scheme

跳转到这个界面

当我设置Build Configuration成Debug时,打印效果图
当我设置Build Configuration成Release的,打印时效果图
当Run Test Profile Analyze Archive的时候,均可以根据须要设置Debug和Release两个模式运行;
因此咱们彻底能够用一个宏指令来设置是否打印调试信息;