我正在研究iOS 7的一些新功能,并使用WWDC视频“在iOS上实现吸引人的UI”中讨论过的一些图像效果。为了在会话的源代码中产生一个模糊的效果,UIImage通过导入UIKit的category进行了扩展:

@import UIKit;

我想我在另一个视频中看到了一些关于这个的东西,但我很难找到它。我在找什么时候使用这个的背景资料。它只能用于苹果框架吗?使用这个编译器指令的好处是否足够让我回去更新旧代码?


当前回答

你可以在《用Objective-C学习可可》(ISBN: 978-1-491-90139-7)这本书中找到很好的答案。

模块是一种将文件和库包含并链接到项目中的新方法。为了理解模块是如何工作的以及它们有什么好处,回顾一下Objective-C的历史和#import语句是很重要的 当你想要包含一个文件使用时,你通常会有一些像这样的代码:

#import "someFile.h"

或者在框架的情况下:

#import <SomeLibrary/SomeFile.h>

Because Objective-C is a superset of the C programming language, the #import state‐ ment is a minor refinement upon C’s #include statement. The #include statement is very simple; it copies everything it finds in the included file into your code during compilation. This can sometimes cause significant problems. For example, imagine you have two header files: SomeFileA.h and SomeFileB.h; SomeFileA.h includes SomeFileB.h, and SomeFileB.h includes SomeFileA.h. This creates a loop, and can confuse the coimpiler. To deal with this, C programmers have to write guards against this type of event from occurring.

当使用#import时,你不需要担心这个问题,也不需要编写头保护来避免它。然而,#import仍然只是一种美化的复制粘贴操作,会导致编译时间变慢,还会出现许多其他较小但仍然非常危险的问题(例如,所包含的文件覆盖了您在自己代码中其他地方声明的内容)。

模块就是为了解决这个问题。它们不再是复制粘贴到源代码中,而是包含的文件的序列化表示,只有在需要的时候和地方才能导入到源代码中。通过使用模块,代码通常会编译得更快,并且比使用#include或#import更安全。

回到前面导入框架的例子:

#import <SomeLibrary/SomeFile.h>

要将这个库作为模块导入,代码将更改为:

@import SomeLibrary;

这有额外的好处,Xcode自动将SomeLibrary框架链接到项目中。模块还允许您只将真正需要的组件包含到项目中。例如,如果你想在AwesomeLibrary框架中使用AwesomeObject组件,通常你必须导入所有的东西来使用其中的一部分。然而,使用模块,你可以只导入你想要使用的特定对象:

@import AwesomeLibrary.AwesomeObject;

对于所有在Xcode 5中创建的新项目,模块默认是启用的。如果你想在旧的项目中使用模块(你真的应该),它们必须在项目的构建设置中启用。一旦你这样做了,你可以在你的代码中同时使用#import和@import语句,而不用担心任何问题。

其他回答

使用模块有几个好处。除非创建了模块映射,否则只能在Apple框架中使用它。@import有点类似于在添加到.pch文件时预编译头文件,这是一种调整应用程序编译过程的方法。此外,您不必以旧的方式添加库,使用@import实际上更快更有效。如果你还在寻找一个好的参考,我强烈推荐你阅读这篇文章。

你可以在《用Objective-C学习可可》(ISBN: 978-1-491-90139-7)这本书中找到很好的答案。

模块是一种将文件和库包含并链接到项目中的新方法。为了理解模块是如何工作的以及它们有什么好处,回顾一下Objective-C的历史和#import语句是很重要的 当你想要包含一个文件使用时,你通常会有一些像这样的代码:

#import "someFile.h"

或者在框架的情况下:

#import <SomeLibrary/SomeFile.h>

Because Objective-C is a superset of the C programming language, the #import state‐ ment is a minor refinement upon C’s #include statement. The #include statement is very simple; it copies everything it finds in the included file into your code during compilation. This can sometimes cause significant problems. For example, imagine you have two header files: SomeFileA.h and SomeFileB.h; SomeFileA.h includes SomeFileB.h, and SomeFileB.h includes SomeFileA.h. This creates a loop, and can confuse the coimpiler. To deal with this, C programmers have to write guards against this type of event from occurring.

当使用#import时,你不需要担心这个问题,也不需要编写头保护来避免它。然而,#import仍然只是一种美化的复制粘贴操作,会导致编译时间变慢,还会出现许多其他较小但仍然非常危险的问题(例如,所包含的文件覆盖了您在自己代码中其他地方声明的内容)。

模块就是为了解决这个问题。它们不再是复制粘贴到源代码中,而是包含的文件的序列化表示,只有在需要的时候和地方才能导入到源代码中。通过使用模块,代码通常会编译得更快,并且比使用#include或#import更安全。

回到前面导入框架的例子:

#import <SomeLibrary/SomeFile.h>

要将这个库作为模块导入,代码将更改为:

@import SomeLibrary;

这有额外的好处,Xcode自动将SomeLibrary框架链接到项目中。模块还允许您只将真正需要的组件包含到项目中。例如,如果你想在AwesomeLibrary框架中使用AwesomeObject组件,通常你必须导入所有的东西来使用其中的一部分。然而,使用模块,你可以只导入你想要使用的特定对象:

@import AwesomeLibrary.AwesomeObject;

对于所有在Xcode 5中创建的新项目,模块默认是启用的。如果你想在旧的项目中使用模块(你真的应该),它们必须在项目的构建设置中启用。一旦你这样做了,你可以在你的代码中同时使用#import和@import语句,而不用担心任何问题。

@import模块(ObjC)或语义导入

而不是通常的模块使用

//as example
#include <Foundation/Foundation.h>
#import <UIKit/UIKit.h>

历史:

[#include -> #import] ->[预编译Headers .pch] -> @import模块(ObjC);-> [import Module(Swift)]

它是LLVM模块的一部分

@ import < module_name >;声明告诉编译器加载(而不是编译)预编译的模块二进制文件,这减少了构建时间。以前编译器编译依赖每次当runt进入它,但现在它应该事先编译,只是加载

//previously
run into dependency -> compile dependency
run into dependency -> compile dependency

//@import
compile dependency 
    run into dependency -> load compiled binary
    run into dependency -> load compiled binary

[Modulemap] -模块和头文件之间的桥梁

Xcode

启用模块(C和Objective-C)(CLANG_ENABLE_MODULES) - clang# include, #import指令自动转换为@import,带来所有优势。Modulemap允许无缝完成,因为它包含了头文件和子/模块之间的映射

当心 - fmodules

#include, #import -> @import

自动链接框架(CLANG_MODULES_AUTOLINK) -允许系统模块自动链接。需要激活CLANG_ENABLE_MODULES。自动链接允许传递-framework <framework_name>基于#import, @import(Objective-C), import(Swift)

如果NO -传递-fno-autolink标志

CLANG_ENABLE_MODULES == NO和CLANG_MODULES_AUTOLINK == NO

如果你想手动处理system(#import <UIKit/UIKit.h>)链接(而不是自动链接),你有两个变量:

在General ->框架和库或框架、库和嵌入式内容中添加依赖项 构建设置->其他连接器标志(OTHER_LDFLAGS) -> -framework <module_name> . Build Settings

下一个错误将被抛出,如果:

Undefined symbol: _OBJC_CLASS_$_UIView

Undefined symbols for architecture x86_64:
  "_OBJC_CLASS_$_UIView", referenced from:
      objc-class-ref in ClassB.o
ld: symbol(s) not found for architecture x86_64
clang: error: linker command failed with exit code 1

CLANG_ENABLE_MODULES被禁用 CLANG_MODULES_AUTOLINK被禁用,没有手动链接

逆向工程

otool -l <binary> 
//-l print the load commands
//find LC_LINKER_OPTION
//cmd LC_LINKER_OPTION

它目前只适用于内置的系统框架。如果你使用#import,就像苹果仍然在应用委托中导入UIKit框架一样,它会被替换(如果modules是打开的,并且它被识别为系统框架),编译器将重新映射它为模块导入,而不是头文件的导入。 因此,在可能的情况下,保留#import将与它转换为模块导入相同

这是一个叫做模块或“语义导入”的新特性。在WWDC 2013的205和404会议视频中有更多的信息。这是预编译头文件的更好实现。你可以在iOS 7和Mavericks中的任何系统框架中使用模块。模块是框架可执行文件及其头文件的打包,被吹捧为比#import更安全、更高效。

使用@import的最大优势之一是你不需要在项目设置中添加框架,它是自动完成的。这意味着您可以跳过单击加号按钮并搜索框架(黄金工具箱)的步骤,然后将其移动到“框架”组。它将使许多开发人员从神秘的“链接器错误”消息中解脱出来。

实际上不需要使用@import关键字。如果你选择使用模块,所有的#import和#include指令都会自动映射到@import。这意味着您不必更改源代码(或从其他地方下载的库的源代码)。据说使用模块也可以提高构建性能,特别是如果您没有很好地使用PCHs,或者如果您的项目有许多小的源文件。

模块是为大多数苹果框架(UIKit, MapKit, GameKit等)预先构建的。你可以在你自己创建的框架中使用它们:如果你在Xcode中创建一个Swift框架,它们会自动创建,你可以手动创建一个“。modulemap”文件自己的任何苹果或第三方库。

你可以使用代码完成来查看可用框架的列表:

在Xcode 5的新项目中,模块默认是启用的。要在旧项目中启用它们,进入您的项目构建设置,搜索“模块”并将“启用模块”设置为“YES”。“Link Frameworks”也应该是“YES”:

你必须使用Xcode 5和iOS 7或Mavericks SDK,但你仍然可以发布旧的操作系统(比如iOS 4.3或其他)。模块不会改变代码的构建方式或任何源代码。


来自WWDC的幻灯片:

导入框架的完整语义描述 不需要解析头文件 导入框架接口的更好方法 加载二进制表示 比预编译头文件更灵活 不受本地宏定义的影响(例如#define readonly 0x01) 默认为新项目启用


要显式地使用模块:

用@import Cocoa替换#import <Cocoa/Cocoa.h>;

你也可以用下面的符号只导入一个头文件:

@import iAd.ADBannerView;

子模块在Xcode中自动完成。