iOS组件化&&二进制&&CocoapodsiOS

iOS-开发进阶07:Module与Swift库

2021-03-10  本文已影响0人  differ_iOSER

iOS 开发进阶 文章汇总

目录

一、Module简介

Module(模块)-最小的代码单元。

假设有A.h、B.h两个头文件、c.m、d.m两个实现文件,两个.m文件都使用#include引入A、B两个头文件。当编译两个.m文件会导致A、B两个头文件分别被编译两次。
为了解决头文件重复编译这个问题现在基本上都使用#import引入头文件,使用#import会默认开启Module,这样头文件会预先编译成二进制,再有文件导入时就不会重新编译。

二、分析Module文件

2.1、通过Module编译代码

准备如下文件:


代码如下:

/* A.h */
#ifdef ENABLE_A
void a() {}
#endif

/* B.h */
#import "A.h"

/* module.modulemap */
module A {
  header "A.h"
}

module B {
  header "B.h"
  export A
}

/* use.c */
#import "B.h"
void use() {
#ifdef ENABLE_A
  a();
#endif
}

build.sh文件代码如下:

# -fmodules:允许使用module语言来表示头文件
# -fmodule-map-file:module map的路径。如不指明默认module.modulemap
# -fmodules-cache-path:编译后的module缓存路径
clang -fmodules -fmodule-map-file=module.modulemap -fmodules-cache-path=../prebuilt -c use.c -o use.o

执行build.sh文件会在prebuilt文件夹中生成两个pcm文件:

两个文件就是预编译好的二进制代码,如果其他文件再引入A和B就不用重新编译了。

2.2、查看AFNetworking文件的modulemap文件
// 声明framework的module名称为AFNetworking
framework module AFNetworking {
  // 导入文件的集合(如果没有关键字header那么umbrella后面需要跟上头文件的文件夹名称)
  umbrella header "AFNetworking-umbrella.h"

  export * //把引入的头文件重新导出。
  module * { export * } //把导入头文件修饰成子module,并把符号全部导出(第一个通配符*表示子module名称和父module名称一致)

// 如果要指定子module的名称需要使用explicit关键字
// eg:
  explicit module NANetworking {
    header "NANetworking.h"
    export *
  }
}

由于我们的项目中会默认开启module,因此无论我们使用#include#import都会自动转变为@import,编译的时候都会被优化成module形式,也就是同一个文件只会被编译一次。

如果希望使用我们自定的module文件,那么需要在Build Setting中设置module map file的路径。

module官方介绍

三、Swift Framework中使用Module

如果我们的Framework中需要用到Swift-OC混编,但是Framework中不能使用桥接文件,因此这种情况下可以使用Module解决。

3.1、创建如下项目文件:

创建NASwiftFrameworkNAOCFramework项目时选择Framework

由于NASwiftFramework中使用了Swift-OC混编,因此编译出现错误,现在我们需要创建Module文件解决这个问题。

3.2、创建NASwiftFramework.modulemap文件(也可以从其他地方CopyCopy时需要勾选Add to targets才能参与编译)

3.3、设置NAOCStudent.h头文件为Public

3.4、设置Module Map File文件路径

现在NASwiftFramework能够编译成功,并且在NAApp项目中也能使用NAOCStudent

3.5、Private Module

如果我们不想直接对外暴漏我们的OC类,我们可以创建NASwiftFramework.private.modulemap

framework module NASwiftFramework_Private { // _Private必须添加,且首字母大写
    module NAOCStudent {
        header "NAOCStudent.h"
        export *
    }
}

然后在Private Module Map File 中指定路径。切换到NASwiftFramework项目进行重新编译

现在NAApp项目中#import <NASwiftFramework/NAOCStudent.h>会报错,但是我们可以通过
@import NASwiftFramework_Private.NAOCStudent;来访问NAOCStudent。如果这一步报如下错误:

Undefined symbols for architecture arm64:
  "_OBJC_CLASS_$_NAOCStudent", referenced from:
      objc-class-ref in ViewController.o
ld: symbol(s) not found for architecture arm64
clang: error: linker command failed with exit code 1 (use -v to see invocation)

需要将NASwiftFramework.framework拖到NAApp项目中

因此Private Module不是真正意义上的私有,只是供开发者区分。如果确实希望隐藏OC代码可以定义相关的协议,Swift通过协议调用OC代码,只对协议进行公开(Build Phases->Headers 中设置协议为PublicOC头文件为Private。如果上面Private Module例子中将NAOCStudent.h设置为Private那么Swift类中也不能使用NAOCStudent)。

四、Swift静态库合并

4.1、Swift头文件

在Xcode 9之后,Swift开始支持静态库。Swift没有头文件的概念,那么我们外界要使用Swift中用Public修饰的类和函数怎么办?

Swift库中引入了一个全新的文件.swiftmodule

.swiftmodule包含序列化过的(AST抽象语法树,Abstract Syntax Tree),也包含SIL (Swift 中间语言,Swift Intermediate Language)。

在上面编译的NASwiftFramework.framework->Show in Finder->Modules->NASwiftFramework.swiftmodule也能看到:

4.2、创建两个Framework库,分别为MySwiftA和MySwiftB

两个库均是静态库并且有一个相同的类

@objc open class MySwiftTeacher: NSObject {
    public func speek() {
        print("speek!")
    }

    @objc public func walk() {
        print("walk!")
    }
}

并把两个静态库编译后的Framework Copy 放到Products目录下(两个项目均添加以下脚本)

cp -Rv -- "${BUILT_PRODUCTS_DIR}/" "${SOURCE_ROOT}/../Products"

编译后结果如下:

合并两个静态库(由于静态库是.o文件的合集,因此合并这两个静态库会产生冲突)

cd Products目录
libtool -static MySwiftA.framework/MySwiftA MySwiftB.framework/MySwiftB -o libMySwiftC.a
//日志警告,两个静态库都包含MySwiftTeacher.o

我们通过ar -t libMySwiftC.a查看libMySwiftC.a中的目标文件

__.SYMDEF
MySwiftA_vers.o
MySwiftTeacher.o
MySwiftB_vers.o
MySwiftTeacher.o

4.3、我们手动组合MySwiftC库

MyApp项目中新建MySwiftC文件夹,并Copy上面生成的相关文件

拖入静态库(勾选Copy item if need):

首次拖入静态库时没有Frameworks文件夹,需要先将静态库拖到General->TARGETS->Frameworks,Libraries,and Embedded Content,然后将Frameworks文件夹中的静态库删除重新拖入并勾选Copy item if need

配置MyApp.Debug.xcconfig文件

HEADER_SEARCH_PATHS = $(inherited) '${SRCROOT}/MySwiftC/Public/MySwiftA.framework/Headers' '${SRCROOT}/MySwiftC/Public/MySwiftB.framework/Headers'

// OTHER_CFLAGS:传递给用来编译C或者OC的编译器,当前就是clang
// -fmodule-map-file: 要加载的module map文件路径
// OC文件中使用静态库需配置如下参数
OTHER_CFLAGS = $(inherited) '-fmodule-map-file=${SRCROOT}/MySwiftC/Public/MySwiftA.framework/module.modulemap' '-fmodule-map-file=${SRCROOT}/MySwiftC/Public/MySwiftB.framework/module.modulemap'

// SWIFT_INCLUDE_PATHS: 传递给SwiftC编译器,告诉他去下面的路径中查找module
// Swift文件中使用静态库需配置如下参数
SWIFT_INCLUDE_PATHS = $(inherited)  '${SRCROOT}/MySwiftC/Public/MySwiftA.framework' '${SRCROOT}/MySwiftC/Public/MySwiftB.framework'

现在MyApp项目中就可以使用静态库了:

ViewController.m

#import "ViewController.h"
#import <MySwiftA-Swift.h>

@interface ViewController ()
@end

@implementation ViewController

- (void)viewDidLoad {
    [super viewDidLoad];
    // Do any additional setup after loading the view.
    
    MySwiftTeacher *t = [MySwiftTeacher new];
}

MySwiftTest.swift

import Foundation
import MySwiftA

@objc open class MySwiftTest: MySwiftTeacher {

    public override init() {
        super.init()
    }
}

五、OC代码映射到Swift的方式

为了让OC代码在Swift使用中做一定的规范,可以进行以下操作。

5.1使用宏

NS_SWIFT_NAME(<#*name#>):给OC方法取别名
NS_TYPED_ENUM:让编译器使用enum
NS_TYPED_EXTENSIBLE_ENUM:让编译器使用Struct
NS_REFINED_FOR_SWIFT 在Swift方法中, 编译器会在名称前加上双下划线__

通过宏配置的弊端:

需要手动修改每个地方的源代码,工作量大

5.2.使用apinotes文件

官方文档
apinotes文件命名规则:前面是项目或者SDK的名称后缀是apinotes
apinotes文件必须放到SDK目录中

---
Name: OCFramework
Classes:
- Name: NAToSwift
  SwiftName: ToSwift  #Swift代码中使用的类名
  Methods:
  - Selector: "changeTeacherName:"
    Parameters:
    - Position: 0
      Nullability: O
    MethodKind: Instance
    SwiftPrivate: true
    # Availability: nonswift   #在Swift中是否可用
    # AvailabilityMsg: "prefer 'deinit'"  #在Swift中不可用的原因
  - Selector: "initWithName:"   #设置其他方法
    MethodKind: Instance
    DesignatedInit: true

六、Module 相关的 Build Setting 参数

6.1对module自身的描述:

6.2对外部module的引用

参考:https://www.jianshu.com/p/d5ca6f0b9ec8

总结

  1. module -> 头文件->目标文件的关系
  2. modulemap ->头文件 -> 目标文件的映射
  3. module:定义一个module
    export :导出当前代表的头文件使用的头文件
    export * :匹配目录下所有的头文件
    module * :目录下所有的头文件都当作一个子module
    explicit : 显式声明一个module的名称
  4. Swift库使用OC代码:不能使用桥接文件
    1. oc的头文件放到modulemap下
    2. oc的头文件放到私有的modulemap下
    3. 协议的方式 投机取巧
  5. Swift静态库的合并
    难点:.swiftmodule 文件(相当于Swift的头文件)
    1. libtool 合并静态库本身
    2. 用到的头文件和Swift头文件和modulemap文件通过目录的形式放到一起
    3. OC要用合并的静态库:clang: other c flags :-fmodule-map-file <modulemap path>
    4. Swift要用合并的静态库 : SwiftC :other swift flags 显式告诉SwiftC <modulemap dir>
  6. OC映射到Swift方式
    1. <工程名称>.apinotes
上一篇 下一篇

猜你喜欢

热点阅读