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分类”中。如果不行的话,就把协议单独放到一个头文件中,然后将其引入

posted @ 2019-10-13 22:41  heil-shikaiming  阅读(289)  评论(0编辑  收藏  举报