第一章: IOS优化(2) @class与引入文件问题

第二条 :
一: 利用@class在类的头文件中可以减少编译时间
  1. 当我们用Xcode创建类时, 其代码看上去如下
//  Person.h
#import <Foundation/Foundation.h>
@interface Person : NSObject
@end
//   Person.m
#import "Person.h"
@implementation Person
@end

用OC编写类几乎都要引入Foundation.h. 如果不在该类中引入这个文件, 那么就要引入与其父类所属框架相对应的"基本头文件", 例如: 在创建IOS应用程序时, 通常会集成UIViewController类, 而这些子类的头文件需要引入UIKit.h, 集成UIViewController哪些类, 可能会用到UIKit中的许多内容

  1. 这段时间如果创建了一个名为Dog的新类, 让每个Person实例都拥有一个Dog. 于是代码如下:
// Person.h
import <Foundation/Foundation.h>
#import "Dog.h"
@interface Person : NSObject
/** Dog */
@property (nonatomic,strong) Dog *dog;
/** name */
@property (nonatomic,strong) NSString *name;
@end
  • 注意: 这么做不够优雅, 在编译使用了一个Person类的文件时, 不需要知道Dog类的全部细节, 只需要知道有一个类名叫Dog就好, 这时, 我们就可以把这种情况告诉编译器, 用@class,代码如下:
//  Person.h
#import <Foundation/Foundation.h>
@class Dog;
@interface Person : NSObject
/** Dog */
@property (nonatomic,strong) Dog *dog;
/** name */
@property (nonatomic,strong) NSString *name;
@end
  • 这叫做"向前声明"(forward declaring)该类
    如果Person的实现文件要使用Dog类, 则必须知道所有接口, 这时实现文件代码就如下:
//  Person.m
#import "Person.h"
#import "Dog.h"
@implementation Person
@end
  • 只有在真的需要时才引入, 这样就可以减少类的使用者所需引入的头文件的数量, 上例中, 把Dog.h引入到Person.h,那么只要引入Dog.h, 就会一并引入Dog.h的所有内容, 这会增加编译时间, 如果使用@class, 则不需要引入很多根本用不到的内容, 这会大大减少编译时间
二: 利用@class可以避免循环引用

1.假设要为Dog类加入新增及删除Master的方法, 那么其头文件中会加入下述方法:

- (void)addMaster:(Person *)person;
- (void)removeMaster:(Person *)person;

这时, 编译器若要编译Dog.h, 那么编译器必须知道Person, 这个类, 而要编译Person.h, 则必须又要知道Dog这个类, 俩个类的声明文件如下(以下情况没有用@class):

//  Person.h
import <Foundation/Foundation.h>
#import "Dog.h"
@interface Person : NSObject
/** dog */
@property (nonatomic,strong) Dog *dog;
@end

// Dog.h  
#import <Foundation/Foundation.h>
#import "Person.h"
@interface Dog : NSObject

- (void)addMaster:(Person *)person;
- (void)removeMaster:(Person *)person;

@end
  • 上面代码会造成, "循环引用"(chicken-and-egg situation), 当解析其中一个头文件时, 编译器会发现它引入了另一个头文件, 而那个头文件又回过头来引用第一个头文件,使用#import而非#include指令虽然不会导致死循环, 这意味着俩个类有一个无法被正确编译, 这时候就需要用到@class, 将其中一个改为使用@class 例如:
//  Dog.h
#import <Foundation/Foundation.h>
@class Person;
@interface Dog : NSObject

  - (void)addMaster:(Person *)person;
  - (void)removeMaster:(Person *)person;

  @end
三: 有时候必须要在头文件中, 引入其他头文件
  1. 如果你写的类, 集成某个类, 则必须引入定义那个父类的头文件
  2. 或者是你声明的类遵从某个协议(protocol), 那么该协议必须有完整定义, 而且不能用向前声明, 向前声明只能告诉编译器有某个协议, 而此时编译器却需要知道该协议中定义的方法, 例如:
    一个Student继承自Person, 并且遵守<SchoolDelegate>协议
#import <Foundation/Foundation.h>
#import "Person.h"
#import "SchoolDelegate.h"
 @interface Student : Person <SchoolDelegate>
 @end
  *  仔细看上述代码就会发现, student类引入了俩个头文件, 其中一个是    SchoolDelegate.h文件, 这个文件里面放存放了协议, 代码如下:  
    ```
     #import <Foundation/Foundation.h>
     @protocol SchoolDelegate <NSObject>

      - (void)buyTicket: (id *)stu;

      @end
     ```
  * 这样将协议单独放到一个文件中虽然也是一种方法, 但是会增加编译时间

  * (delegate protocol)这时候可以将协议与接收协议的类放在一起定义, 最好能在实现文件中声明此类实现了该代理方法, 并把实现代码放在"分类", 这样的话, 只要在实现文件中引入包含协议的头文件即可, 不需要将这个文件放在公共头文件里, 代码如下:
          //Student.m  文件
          #import "Student.h"

          @protocol SchoolDelegate <NSObject>
          - (void)buyTicket:(id)stu;
          @end

          @implementation Student
          @end
重点:
  1. 每次在头文件中引入其他头文件之前, 都要先问问自己这样做是否有必要, 如果可以用向前声明取代引入, 那么最好用向前声明, 若因为实现属性, 实例变量, 或者要遵守协议而必须引入头文件, 则尽量将其移到"class-continuation分类"中
  2. 这样做的好处, 不近可以缩减编译时间, 而且还能降低彼此依赖程序,
  3. 除非有必要, 否则不要引入头文件, 一般再来, 应在某个类的头文件中使用向前声明来提及别的类, 并在实现文件中引入那些类的头文件, 这样做可以降低耦合
  4. 有时无法使用向前声明, 比如遵守协议, 这种情况下, 尽量把"该类遵循某协议"的这条声明移到"class-continuation分类"中, 如果不行的话, 就把协议单独放在一个头文件

声明: 以上大部分内容均来自 Effective Objective-C

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 202,980评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,178评论 2 380
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 149,868评论 0 336
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,498评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,492评论 5 364
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,521评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,910评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,569评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,793评论 1 296
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,559评论 2 319
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,639评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,342评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,931评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,904评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,144评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,833评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,350评论 2 342

推荐阅读更多精彩内容

  • 1.项目经验 2.基础问题 3.指南认识 4.解决思路 ios开发三大块: 1.Oc基础 2.CocoaTouch...
    阳光的大男孩儿阅读 4,967评论 0 13
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,580评论 18 139
  • Objective-C 1. import的用法 拷贝文件内容可以自动防止文件的内容被重复拷贝(#define宏定...
    马文涛阅读 5,326评论 3 17
  • 我先问大家一个问题:"是长枪的射程远,还是短枪的射程远?还有两者谁的精准度更好"。 从我们看过的电影来说...
    一匹马来了阅读 1,008评论 0 0
  • 又是一个起风的夜晚 卸去一天疲惫的伪装 把你的倩影在心里放大一张又一张 所有的霓虹都比不上你眼睛的明亮 分手的时候...
    坐看云起的Annie阅读 475评论 7 12