1. 全局的 native module 注册表
rctmoduleclasses 数组
首先, rn中拥有一个全局的静态数组rctmoduleclasses, 使用它来记录所有的native module的class, 可以认为这是一个待启动的native module配置表!!!
并且提供了一个全局方法void rctregistermodule(class);, 向这个rctmoduleclasses注入新的module class:
// 1. 全局注册中心, rn 用 rctmoduleclasses 来持有全局注册的 native module
static nsmutablearray<class> *rctmoduleclasses;
static dispatch_queue_t rctmoduleclassessyncqueue;
// 2. sync 读 - 全局 rctmoduleclasses
nsarray<class> *rctgetmoduleclasses(void) {
__block nsarray<class> *result;
dispatch_sync(rctmoduleclassessyncqueue, ^{
result = [rctmoduleclasses copy];
});
return result;
}
// 3. barrier async 写
void rctregistermodule(class moduleclass) {
static dispatch_once_t oncetoken;
dispatch_once(&oncetoken, ^{
rctmoduleclasses = [nsmutablearray new];
rctmoduleclassessyncqueue = dispatch_queue_create("com.facebook.react.moduleclassessyncqueue", dispatch_queue_concurrent); // sync 读, barrier_async 写
});
dispatch_barrier_async(rctmoduleclassessyncqueue, ^{
[rctmoduleclasses addobject:moduleclass];
});
}
2. rctbridgemodule 协议
app在pre main将module注入全局表中
rn中如果要实现一个native module, 需要实现rctbridgemodule协议, 并且rn官方要求我们在实现协议时, 必须加入rct_export_module宏.
@protocol rctbridgemodule <nsobject>
/**
* place this macro in your class implementation to automatically register
* your module with the bridge when it loads. the optional js_name argument
* will be used as the js module name. if omitted, the js module name will
* match the objective-c class name.
*/
#define rct_export_module(js_name) \
rct_extern void rctregistermodule(class); \
+(nsstring *)modulename \
{ \
return @ #js_name; \
} \
+(void)load \
{ \
rctregistermodule(self); \
}
...
@end
其中rct_export_module(js_name)宏的实现中, 会在+load中调用rctregistermodule(self); 方法, 将native module class注入到全局的静态数组rctmoduleclasses中!!!
也就是说, ios app在启动执行main函数之前, 所有的实现rctbridgemodule协议的类型class, 都会被注入到rctmoduleclasses这个数组中.
3. rctmoduleclasses中class数据的处理
rctcxxbridge的start方法执行时, rctgetmoduleclasses()方法里面已经拥有了所有需要注册到bridge的native module配置表, 因此直接可以对这个module配置表中, 所有的 class进行初始化!!!
//(void)[self _initializemodules:rctgetmoduleclasses() withdispatchgroup:preparebridge lazilydiscovered:no];
- (nsarray<rctmoduledata *> *)_initializemodules:(nsarray<class> *)modules
withdispatchgroup:(dispatch_group_t)dispatchgroup
lazilydiscovered:(bool)lazilydiscovered {
/// 1. modules class => 包装成中间管理类 rctmoduledata
/// 2. 然后将 rctmoduledata 对象放到 bridge 的几个特定容器中管理
nsarray<rctmoduledata *> *moduledatabyid = [self _registermodulesforclasses:modules lazilydiscovered:lazilydiscovered];
if (lazilydiscovered) {
...
} else {
// 初始化时候, 会走这里
/// 3. 处理 moduledata.hasinstance 的场景, 在 bridge._modulesetupcomplete 之前, module强制进行instance实例化, 并管理在 moduledata 中
for (rctmoduledata *moduledata in _moduledatabyid) {
if (moduledata.hasinstance && (!moduledata.requiresmainqueuesetup || rctismainqueue())) {
(void)[moduledata instance];
}
}
/// 4. 标记 bridge 中的 module 完成 setup 工作
_modulesetupcomplete = yes;
/// 5. 对 bridge 中的 module 异步 preapre
[self _preparemoduleswithdispatchgroup:dispatchgroup];
}
/// 6. 返回所有的 moduledata 数组
return moduledatabyid;
}
主要来说是做以下几个事情:
- _registermodulesforclasses, 主要工作: module class包装成rctmoduledata:
将每个moduleclass封装成一个个rctmoduledata 对象moduledata
然后对moduledata进行setup()
- 将rctmoduledata注册到bridge的module管理的属性中:
nsmutabledictionary<nsstring *, rctmoduledata *> *_moduledatabyname
nsmutablearray<rctmoduledata *> *_moduledatabyid;
nsmutablearray<class> *_moduleclassesbyid;
- 部分rctmoduledata模块需要在bridge的_modulesetupcomplete = true之前就进行[moduledata instance] -- (void)[moduledata instance];
- 标记bridge的_modulesetupcomplete, 标记 bridge的 module setup 完成!!!
- 异步!!! 大部分的rctmoduledata可以等到bridge完全初始化以后进rctmoduledata instance -- 这类module在实例化时, 会进入调度组dispatchgroup中, 然后异步到mainqueue进行(void)[moduledata instance];
这里有一个重要的标记, rctcxxbridge.modulesetupcomplete = true, 标记 bridge 的native module setup完成!!!
异步: 大部分的moduledata instance 都会通过dispatchgroup方式, 异步进行[moduledata instance]
4. moduleclasse包装成rctmoduledata过程
我们知道一个native module被rn bridge使用, 会经历两个过程
- module class 成员方法的解析
- module instnace过程
先看第一个, 从class包装成rctmoduledata时, 发生了什么 1. initwithmoduleclass 2. setup: 帮助判断class中的一些实例方法和大量配置
- (instancetype)initwithmoduleclass:(class)moduleclass
moduleprovider:(rctbridgemoduleprovider)moduleprovider
bridge:(rctbridge *)bridge
moduleregistry:(rctmoduleregistry *)moduleregistry
viewregistry_deprecated:(rctviewregistry *)viewregistry_deprecated
bundlemanager:(rctbundlemanager *)bundlemanager
callablejsmodules:(rctcallablejsmodules *)callablejsmodules
{
if (self = [super init]) {
_bridge = bridge;
_moduleclass = moduleclass;
_moduleprovider = [moduleprovider copy];
_moduleregistry = moduleregistry;
_viewregistry_deprecated = viewregistry_deprecated;
_bundlemanager = bundlemanager;
_callablejsmodules = callablejsmodules;
[self setup];
}
return self;
}
- (void)setup {
// 1. instance 是否实现 -batchdidcomplete 方法
_implementsbatchdidcomplete = [_moduleclass instancesrespondtoselector:@selector(batchdidcomplete)];
// 2. instance 是否实现 -batchdidcomplete 方法
_implementspartialbatchdidflush = [_moduleclass instancesrespondtoselector:@selector(partialbatchdidflush)];
// 3. instance 是否是用常量需要暴露出去
_hasconstantstoexport = [_moduleclass instancesrespondtoselector:@selector(constantstoexport)];
// 4. module 是否强制在 main queue 中 setup
const bool implementsrequiremainqueuesetup = [_moduleclass respondstoselector:@selector(requiresmainqueuesetup)];
if (implementsrequiremainqueuesetup) {
_requiresmainqueuesetup = [_moduleclass requiresmainqueuesetup];
} else {
...
// 5. 没实现 requires 时,
// - 是否有自定义 -init 初始化方法
// - _hasconstantstoexport || hascustominit 时, 需要main queue setup
const bool hascustominit = !_instance && [_moduleclass instancemethodforselector:@selector(init)] != objectinitmethod;
_requiresmainqueuesetup = _hasconstantstoexport || hascustominit;
}
}
其中比较关键的是通过rctmoduledata描述module class的几个关键属性:
- _implementsbatchdidcomplete - 标记 instance 是否实现关键方法
- _implementspartialbatchdidflush - 标记 instance 是否实现关键方法
- _hasconstantstoexport - 标记 module 是否有 constants dictionary 暴露给 js
- _requiresmainqueuesetup - 注意在_preparemoduleswithdispatchgroup中会使用, 标记 module instance setup 是, 是否强制在main queue中调用. rn默认会在子线程中进行module instance setup
5. rctmoduledata在什么时候进行module instance
native module在真正被js使用之前, 需要对rctmoduledata进行模块实例化 -- module instnace, 也称为module instance setup, 前面提到过, 大量的rctmoduledata的module instnace过程会在rctcxxbridge._preparemoduleswithdispatchgroup(...)方法中实现:
/// rctcxxbridge 处理所有的 rctmoduledata, 对它们调用 `moduledata instance` 方法的过程
- (void)_preparemoduleswithdispatchgroup:(dispatch_group_t)dispatchgroup {
// 1. 根据是否有 dispatchgroup 决定后续进行 `module instance` 是否异步进行
bool initializeimmediately = no;
if (dispatchgroup == null) {
rctassertmainqueue();
initializeimmediately = yes;
}
// 2. _moduledatabyid 中缓存着所有的`module class`构造的`rctmoduledata`, 遍历它们, 构造一个 block, 根据 initializeimmediately 参数, 决定是否立即执行
for (rctmoduledata *moduledata in _moduledatabyid) {
// 3. 注意, 强制 moduledata.requiresmainqueuesetup, 也就是`module instance`强制在main queue 进行`setup`
if (moduledata.requiresmainqueuesetup) {
// 4. `module instance setup` 的过程, 直接主动调用 [moduledata instance], 并强制触发 `[moduledata gatherconstants]`, 收集 instance 要暴露给 js 的常量dictionary
dispatch_block_t block = ^{
if (self.valid && ![moduledata.moduleclass issubclassofclass:[rctcxxmodule class]]) {
(void)[moduledata instance];
if (!rctismainqueueexecutionofconstantstoexportdisabled()) {
[moduledata gatherconstants];
}
}
};
if (initializeimmediately && rctismainqueue()) {
block();
} else {
if (dispatchgroup) {
dispatch_group_async(dispatchgroup, dispatch_get_main_queue(), block);
}
}
// 5. rctcxxbridge 中 记录所有在 main queue 中 进行 `module instance` 的module个数
_modulesinitializedonmainqueue++;
}
}
}
通过上面的代码, 我们能看到: 只有moduledata.requiresmainqueuesetup == true时, 才会在启动阶段进行module instance, 如果没有强制要求在主线程进行module instance setup那么就是 lazy module instance, 或者称为懒实例化的模块!!!
通过前面的rn的代码, 我们能看到, 有以下几种场景, module instance被强制在启动阶段被module instance:
- 在实现rctbridgemodule协议时, +requiresmainqueuesetup 方法返回 true
- 如果没有实现+requiresmainqueuesetup方法, 判断 hascustominit || _hasconstantstoexport, 也就是如果有 自定义-init方法, 或者 constants暴露给js, 必须强制在 main queue setup
5. rctmoduledata在进行module instance的细节
直接贴上 rctmoduledata的instance方法, 其中最重要的是方法-setupinstanceandbridge:
- (id<rctbridgemodule>)instance {
...
// 1. _setupcomplete 标记 module instance 是否 setup 完成
if(!_setupcomplete) {
[self setupinstanceandbridge:requestid];
}
...
return _instance;
}
- (void)setupinstanceandbridge:(int32_t)requestid {
nsstring *modulename = [self name];
// 注意: 使用 instancelock 保护 _instance 成员
{
std::unique_lock<std::mutex> lock(_instancelock);
// 1. 判断 moduledata._setupcomplete 是否完成 setup
bool shouldsetup = !_setupcomplete && _bridge.valid;
// 2. 如果 moduledata._instance 没实例化, 使用 _moduleprovider() 实例化
// - 如果实例化失败, 也需要标记 _setupcomplete = true
if (shouldsetup) {
if (!_instance) {
// 使用 moduleprovider() 调用实例化
_instance = _moduleprovider ? _moduleprovider() : nil;
if (!_instance) {
_setupcomplete = yes;
}
}
}
// 3. 将常见属性(bridge, moduleregistry...)注入到 module instnace 中!
if (shouldsetup) {
[self setbridgeforinstance];
[self setmoduleregistryforinstance];
[self setviewregistryforinstance];
[self setbundlemanagerforinstance];
[self setcallablejsmodulesforinstance];
}
// 4. 初始化 module instance 的 methodqueue
[self setupmethodqueue];
// 5. 调用 module instance 中的 initialize 方法, 如果实现了的话
if (shouldsetup) {
[self _initializemodule];
}
} // instancelock 释放
// 6. 什么时候通知全局 module instance 完成!!!
if (_bridge.modulesetupcomplete) {
// 6.1 大部分 module instance 是在 modulesetupcomplete = ture 执行, 会主动通知全局
[self finishsetupforinstance];
} else {
// 6.2 少部分在 modulesetupcomplete = false 执行, 标记 _requiresmainqueuesetup = no, 这样 module instance 实际只setup 一半, lazy instance 使用时, `_bridge.modulesetupcomplete` 一定为 true, 再进行通知, 调用 `finishsetupforinstance`方法
_requiresmainqueuesetup = no;
}
}
- (void)setupmethodqueue {
if (_instance && !_methodqueue && _bridge.valid) {
// 1. instance 是否主动指定 methodqueue, moduledata持有
bool implementsmethodqueue = [_instance respondstoselector:@selector(methodqueue)];
if (implementsmethodqueue && _bridge.valid) {
_methodqueue = _instance.methodqueue;
}
// 2. instance 没有指定, 创建 子线程, 作为 methodqueue
if (!_methodqueue && _bridge.valid) {
_queuename = [nsstring stringwithformat:@"com.facebook.react.%@queue", self.name];
_methodqueue = dispatch_queue_create(_queuename.utf8string, dispatch_queue_serial);
// 3. 如果`module instance` 主动实现 methodqueue, 使用kvc设置给`module instance` methodqueue!!!
if (implementsmethodqueue) {
@try {
[(id)_instance setvalue:_methodqueue forkey:@"methodqueue"];
} @catch (nsexception *exception) {
rctlogerror();
}
}
}
}
// 如果 `module instance` 实现 `rctbridgemodule`协议的`-initialize`方法, 主动调用, 并标记`_isinitialized`执行过
- (void)_initializemodule {
if (!_isinitialized && [_instance respondstoselector:@selector(initialize)]) {
_isinitialized = yes;
[(id<rctinitializing>)_instance initialize];
}
}
// 如果执行这个方法, 通知bridge + 全局: 这个 module `setupcomplete`!!!
- (void)finishsetupforinstance {
if (!_setupcomplete && _instance) {
_setupcomplete = yes;
[_bridge registermoduleforframeupdates:_instance withmoduledata:self];
[[nsnotificationcenter defaultcenter] postnotificationname:rctdidinitializemodulenotification object:_bridge userinfo:@{@"module" : _instance, @"bridge" : rctnullifnil(_bridge.parentbridge)}];
}
}
在module instance过程中, 有以下几个点需要注意:
- instance = _moduleprovider(), 而 _moduleprovider实现基本都是[moduleclass new], 也就是说使用-init进行初始化.
- 能看给module instance注入大量属性(bridge, moduelregistry...)时, 都是使用的 kvc, 并包装在try-catch中, 因为rctbridgemodule协议实现了接口, 如果 module需要使用这些注入的 api, 需要手动使用@synthesize生成对应的成员变量.
- 关于method queue, 后续 js 调用 module instance的export method时, 会异步到method queue中执行.
- 调用module instance的-initialize是rctbridgemodule协议提供的, 请与oc中的+initiali...进行区分
关于finishsetupforinstance的调用时机, 请参考代码注释. 另外, 只要它调用会进行如下操作:
- 标记 _setupcomplete = yes
- 需要通知并更新, bridge中frameupdates相关的module
- 全局通知rctdidinitializemodulenotification
以上就是ios rn启动中管理native module详解的详细内容,更多关于ios rn启动管理native module的资料请关注代码网其它相关文章!
发表评论