52个有效方法(2) - 在类的头文件中尽量少引入其他头文件

"头文件" 与 "实现文件"
  • Objective-C语言编写"类"的标注方式为: 以类名做文件名, 分别创建两个文件, 头文件后缀用.h, 实现文件后缀用.m。
#include指令
  • #include 是C语言的预处理指令之一,所谓预处理,就是在编译之前做的处理。预处理指令一般以 # 开头

  • #include 指令后面会跟着一个文件名,预处理器发现 #include 指令后,就会根据文件名去查找文件,并把这个文件的内容包含到当前文件中。

  • 被包含文件中的文本将替换源文件中的 #include指令,就像你把被包含文件中的全部内容拷贝到这个 #include 指令所在的位置一样。

  • 如果要想使用声明在.h头文件 的函数,就必须先用 #include 指令包含函数所在的.h头文件。

  • #include 指令不仅仅限于.h头文件,可以包含任何编译器能识别的C/C++代码文件,包括.c、.hpp、.cpp等,甚至.txt、.abc等等都可以。

#include""#include<>
  • 当包含我们自己写的文件就是使用 #include "";当包含系统􏰀提供的头文件的时候,就是用#include <>注意:include 语句之后不需要加";"(因为#include它为一个预处理指令,不是一个语句)

  • #include""#include<>二者的区别在于:当被include的文件路径不是绝对路径的时候,有不同的搜索顺序。

    • #include""搜索顺序:

    1.先在这条include指令的父文件所在文件夹内搜索,所谓的父文件,就是这条include指令所在的文件;
    2.如果上一步找不到,则在父文件的父文件所在文件夹内搜索;
    3.如果上一步找不到,则在编译器设置的include路径内搜索;
    4.如果上一步找不到,则在系统的include环境变量内搜索。

    • #include<>搜索顺序:

      在编译器设置的include路径内搜索;
      如果上一步找不到,则在系统的include环境变量内搜索。

#import指令
  • #import#include的类似,都是把其后面的文件拷贝到该指令所在的地方

  • #import具有防止重复引用的功能。在使用C语言时, 通常使用宏定义防止重复引用。

  • #import <> 用于包含系统文件,#import""用于包含本项目中的文件。

@import ClassName 的使用
  • #import导入程序所需的文件越多,所需的编译时间越长,而有大量重复。可创建.pch文件解决编译时间长的问题。但是维护.pch文件是很低效的。

  • 使用 @import指令可使编译器优化预编译头文件和缓存编译结果。同时,文件中不用再明确引用框架--编译器会根据@import自动导入相应框架。

  • 但该指令现阶段还只能用于Apple提供的框架,如果导入的是自己的类或框架还是需要#import

@class ClassName 的使用
  • 在头文件中声明某个类的对象时, 需要告诉编译器此类存在, 但又不需要知道类的全部细节。此时可以使用 @class ClassName 告诉编译器。这叫做: “向前声明”。(如果你写的类继承自某个超类, 则必须引入定义那个超类的头文件)

  • 在实现文件中则需要引入类的头文件, 因为在实现文件中需要使用类的细节。

  • 注意: 将引入头文件的时机尽量延后, 只有在确有需要时才引入, 这样就可以减少类的使用者所引入头文件数量。以减少程序的编译时间。

@class 和"循环引用"
  • 类A与类B,如果在各自头文件中引入对方的头文件,则会导致“循环引用”。当解析其中一个头文件时,编译器会发现它引入了另一个头文件,而那个头文件又回过头来引用第一个头文件。

  • 使用#import而非#include指令虽然不会导致死循环,但却意味着两个类里有一个无法被正确编译。

  • 使用@class可以避免这种"循环引用"的问题

@class 和协议
  • 如果要声明某个类遵循一项协议, 那么该协议必须有完整定义, 且不能使用向前声明。

  • 向前声明只能告诉编译器有某个协议, 而此时编译器却要知道协议中定义的方法。

  • 这种情况下,尽量把“该类遵循某协议”的这条声明移至“class-continuation分类”中。

"class-continuation分类"和普通的分类不同,它必须定义在其所接续的那个类的实现文件里。其重要之处在于,这是唯一能声明的实例变量的分类,而且此分类没有特定的实现文件,其中的方法都应该定义在类的主实现文件里(.m文件中)。
  • 如果不行的话,就把协议单独放到一个头文件中,然后将其引入。
要点:
  1. 除非确有必要,否则不要引入头文件。一般来说,应在某个类的头文件中使用向前声明(就是@class obj)来提及别的类,并在实现文件中引入那些类的头文件。这样可以尽量降低类之间的耦合(coupling) (在.h文件中使用@class obj来声明,在.m文件中再使用#import来引入头文件)

  2. 有时无法使用向前声明,比如要声明某个类遵循一项协议。这种情况下,尽量把“该类遵循某协议”的这条声明移至“class-continuation分类”中。如果不行的话,就把协议单独放到一个头文件中,然后将其引入

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

推荐阅读更多精彩内容