modulemap的作用
 
modulemap 文件是用来解决 C,Object-C,C++ 代码在 Swift 项目中集成的问题的。
在 Swift 项目中,如果需要使用 C,Object-C 或者 C++ 代码,需要将相应的头文件和源文件导入到项目中,并且需要手动管理它们之间的依赖关系。导致项目结构复杂,管理困难。
为了解决这个问题,可以使用 modulemap 文件来定义模块,将 C,Object-C 或者 C++ 代码组合成一个模块,然后在 Swift 项目中直接导入该模块。

文档中的说明:

As Bridging-Header can help us in App Target and App Test Target, not in static library or dynamic libraries to use the Objective C / C APIs into Swift classes, modulemap can help us here.
 
在Swift项目中使用Objective C / C APIs的方式
Swift使用Objective C / C源码时:在App工程中OC文件可以通过放置在Bridging-Header文件中让Swift其他文件引用。
Swift使用Objective C / C的静态库,动态库时:在framework静态库,动态库中OC文件可以通过给framework添加module.modulemap让Swift通过“import xxx模块”方式对文件进行引用的。
Module导入方式优点
 
传统.h文件导入的问题
对于基于C语言而来的其他语言,在导入时,都是导入.h文件,开发者目测很难区分到底导入的是C,C++,还是OC
编译性能问题
编译器在预编译阶段碰到import xx.h后,会将xx文件复制,替换到xx.h这个位置。如果一个.h文件中包含了多个其他.h文件,其他.h文件中又相互包含时,则会出现相同的代码多次被替换。
另外模块多次导入时,还易出现宏定义替换不完全,错误替换的问题(如:当前文件有方法A,它包含的文件中存在了宏定义A,那改如果替换呢)
module导入方式
每个模块都是一个完全隔离的个体。
当模块第一次被import时,编译器会根据modulemap把它里面的模块编译成预编译模块(Precompiled Module)pcm文件,并将其在本地缓存,里面包含了模块的所有API信息。
当第二个模块被import导入时,编译器会直接找这个模块被编译后的缓存二进制文件。提升了编译效率。
Module模块的使用方式
普通导入方式:
#import <MyModule/MyModule-Swift.h>
模块导入语法: 
@import MyModule;
Swift导入模块的方式
import MyModule
使用模块导入方式就需要framework下包含module.modulemap文件,modulemap指明了framework中的头文件的逻辑结构和如何映射成模块。
在使用 MyModule 模块时,就可以直接导入MyModule模块文件,而不需要手动逐个添加里面的子模块。
Module模块语法
//给Swift项目用的
framework module MyModule {
umbrella header "MyModule.h" // headers.h 和 module.modulemap 必须在同一group下,否则需要配置 `header "/??/headers.h"`
header "headers.h" requires objc
export * //将子模块都导出到主模块
module * { export * } //将当前目录下所有的头文件(包括umbrella header中包含的每个头文件和其他header "headers.h"头文件)都导出成子模块
} //给OC项目用的,需要支持objc语言环境
module MyModule.Swift {
header "MyModule-Swift.h"
requires objc
}
module * { export * } :将当前目录下所有的头文件打包成一个模块,
export * 表示:将其他模块中的所有头文件都导出到当前模块中
export * 和 module * { export * } 同时使用表示将当前目录下所有的头文件打包成一个模块,并将其他模块中的所有头文件也导出到当前模块中。
header 命令:表示将指定的头文件添加到当前模块中。
umbrella header 命令:表示将指定的头文件视为一个 umbrella header,该头文件中包含了其他多个头文件的接口。这个头文件中包含了其他多个头文件的接口,因此可以使用 export * 命令将所有接口都导出到当前模块中。
framework module XXXX 定义了一个 framework 语义的模块
requires objc 说明:导入模块的编译单元要支持OC语言环境
header "headers.h" 说明:将头文件aa.h映射为模块
 
模块声明
[framework] module module-id [extern_c] [system] {
module-member
}
extern_c:表示moduel中的C代码可以被C++使用,相当于添加了extern 'C'这个声明。
 
常见Module目录结构
Name.framework/
Modules/module.modulemap framework 的模块映射
Headers/ 包含了 framework 中的头文件
PrivateHeaders/ 包含了 framework 中私有的头文件
Frameworks/ 包含嵌入的其它 framework
Resources/ 包含额外的资源
Name 指向共享库的符号链接
 
另外
Xcode创建的APP和库默认都是支持Moduel导入的,如果不支持可以手动在在 Build Settings 中,Defines Module 的设置为 YES,进行支持。
参考文章:
https://juejin.cn/post/7139724115157450765
https://zhuanlan.zhihu.com/p/602783297
https://www.jianshu.com/p/ce49d8f32f77

modulemap的使用方法的更多相关文章

  1. 用swift开发framework时采用OC混编的解决方案

    随着swift ABI的稳定,越来越多的开发者开始使用swift语言开发项目,但是由于大部分工具库也还是使用OC写的,因此我们不得不需要在项目中采用swift与oc混编. 在开发app项目时,swif ...

  2. javaSE27天复习总结

    JAVA学习总结    2 第一天    2 1:计算机概述(了解)    2 (1)计算机    2 (2)计算机硬件    2 (3)计算机软件    2 (4)软件开发(理解)    2 (5) ...

  3. mapreduce多文件输出的两方法

    mapreduce多文件输出的两方法   package duogemap;   import java.io.IOException;   import org.apache.hadoop.conf ...

  4. 【.net 深呼吸】细说CodeDom(6):方法参数

    本文老周就给大伙伴们介绍一下方法参数代码的生成. 在开始之前,先补充一下上一篇烂文的内容.在上一篇文章中,老周检讨了 MemberAttributes 枚举的用法,老周此前误以为该枚举不能进行按位操作 ...

  5. IE6、7下html标签间存在空白符,导致渲染后占用多余空白位置的原因及解决方法

    直接上图:原因:该div包含的内容是靠后台进行print操作,输出的.如果没有输出任何内容,浏览器会默认给该空白区域添加空白符.在IE6.7下,浏览器解析渲染时,会认为空白符也是占位置的,默认其具有字 ...

  6. 多线程爬坑之路-Thread和Runable源码解析之基本方法的运用实例

    前面的文章:多线程爬坑之路-学习多线程需要来了解哪些东西?(concurrent并发包的数据结构和线程池,Locks锁,Atomic原子类) 多线程爬坑之路-Thread和Runable源码解析 前面 ...

  7. [C#] C# 基础回顾 - 匿名方法

    C# 基础回顾 - 匿名方法 目录 简介 匿名方法的参数使用范围 委托示例 简介 在 C# 2.0 之前的版本中,我们创建委托的唯一形式 -- 命名方法. 而 C# 2.0 -- 引进了匿名方法,在 ...

  8. ArcGIS 10.0紧凑型切片读写方法

    首先介绍一下ArcGIS10.0的缓存机制: 切片方案 切片方案包括缓存的比例级别.切片尺寸和切片原点.这些属性定义缓存边界的存在位置,在某些客户端中叠加缓存时匹配这些属性十分重要.图像格式和抗锯齿等 ...

  9. [BOT] 一种android中实现“圆角矩形”的方法

    内容简介 文章介绍ImageView(方法也可以应用到其它View)圆角矩形(包括圆形)的一种实现方式,四个角可以分别指定为圆角.思路是利用"Xfermode + Path"来进行 ...

  10. JS 判断数据类型的三种方法

    说到数据类型,我们先理一下JavaScript中常见的几种数据类型: 基本类型:string,number,boolean 特殊类型:undefined,null 引用类型:Object,Functi ...

随机推荐

  1. 介绍一个.Net远程日志组件

    对于软件开发的阶段和正式运行阶段,我们都需要查看日志来诊断出现的问题.不过,在查看日志时需要登录服务器,找到特定的日志文件,再查看其中的内容,这显然不是很方便. 为了解决这个问题,我们可以使用远程日志 ...

  2. Python实现网络工具

    使用python编写网络工具 基础内容 介绍基本的网络编程 Socket编程 Socket又称"套接字",应用程序通常通过"套接字"向网络发出请求或者应答网络请 ...

  3. 详解Redis三大集群模式,轻松实现高可用!

    1. Redis集群简介 1.1 什么是Redis集群 Redis集群是一种通过将多个Redis节点连接在一起以实现高可用性.数据分片和负载均衡的技术.它允许Redis在不同节点上同时提供服务,提高整 ...

  4. 数据库mysql8.0安装+Navicat12

    第一次写博客,准备坚持下去把每天学习到的东西记录下来,刚好安装mysql8.0过程中遇到了一些问题并解决,系统win10.致敬博客 首先先粘贴一段mysql的百度百科词条:MySQL是一个关系型数据库 ...

  5. Docker快速部署Hadoop环境

    文章目录 安装环境 安装过程 拉取镜像 在Docker中创建网络,方便通信 创建Master节点 创建slave1和slave2节点 分别进入三个容器修改hosts文件 在Master执行集群初始化 ...

  6. java镜子之反射篇

    文章目录 注解 内置注解 元注解 反射 类的初始化 类加载器 双亲委派机制 反射方法的使用 调用类的方法.成员变量.构造器等 总结 注解和反射是Java中非常重要的知识,一些优秀开源的框架都是大量运用 ...

  7. 【H5】Emmet 指令 HTML

    Emmet操作指南 HTML篇 生成带有内容的标签 标签名{内容}可以生成带有内容的标签 div{abc} <div>abc</div> 生成带有属性的标签 生成带有class ...

  8. Layui+dtree实现左边分类列表,右边数据列表

    效果如下 代码实现 <!DOCTYPE html> <html> <head> <meta charset="utf-8"> < ...

  9. AIGC持续火爆大模型争相推出,庞大市场造就算力供应模式演变

    本图由AI生成 黄仁勋说的AI发展迎来iPhone时刻,对NVIDIA有什么影响? 文/王吉伟 近期的AIGC领域仍旧火爆异常. 但火的不只是AIGC应用,还有巨头之间的AI竞赛,以及接连不断上新的A ...

  10. 整合vxgPlayer使chrome支持vxg_media_player播放rtsp视频

    目前有一个关于接入海康监控进行视频融合的项目需求,按理说在前端技术发展如此迅速的今天,使用web播放一个视频应该是不算什么难事,只是万事都有意外,因很多视频厂家的监控数据都不是普通的mp4啥的,所以使 ...