ReactNative 通信机制_c++端源码分析

ReactNative: 源码分析系列

ReactNative 启动过程源码分析
ReactNative 通信机制_java端源码分析
ReactNative 通信机制_c++端源码分析
ReactNative 通信机制_js端源码分析

上一章我们分析了java端通信机制流程,这章我们分析c++端调用流程。

CatalystInstanceImpl.cpp

对应CatalystInstanceImpl.java类,它是java端native方法的具体实现。

void CatalystInstanceImpl::initializeBridge(
    jni::alias_ref callback,
    // This executor is actually a factory holder.
    JavaScriptExecutorHolder* jseh,
    jni::alias_ref jsQueue,
    jni::alias_ref nativeModulesQueue,
    jni::alias_ref uiBackgroundQueue,
    jni::alias_ref
        ::javaobject> javaModules,
    jni::alias_ref
        ::javaobject> cxxModules) {
  moduleMessageQueue_ = std::make_shared(nativeModulesQueue);
  if (uiBackgroundQueue.get() != nullptr) {
    uiBackgroundMessageQueue_ = 
      std::make_shared(uiBackgroundQueue);
  }

  moduleRegistry_ = std::make_shared(
    buildNativeModuleList(
       std::weak_ptr(instance_),
       javaModules,
       cxxModules,
       moduleMessageQueue_,
       uiBackgroundMessageQueue_));

  instance_->initializeBridge(
    folly::make_unique(
    callback,
    uiBackgroundMessageQueue_ != 
          NULL ? uiBackgroundMessageQueue_ : moduleMessageQueue_),
    jseh->getExecutorFactory(),
    folly::make_unique(jsQueue),
    moduleRegistry_);
}
  1. 创建一个ModuleRegistry实例, buildNativeModuleList是ModuleRegistryBuilder.cpp中的方法,用来将javaModules和cxxModules这些java实例集合转换成c++的NativeModule实例集合。
    2.调用instance_的initializeBridge方法。 instance_是instance.cpp的实例。
void CatalystInstanceImpl::jniCallJSFunction(std::string module, 
        std::string method, NativeArray* arguments) {
  instance_->callJSFunction(std::move(module),
                            std::move(method),
                            arguments->consume());
}

只是回调instance_对应方法。

void CatalystInstanceImpl::jniCallJSCallback(jint callbackId,
         NativeArray* arguments) {
  instance_->callJSCallback(callbackId, arguments->consume());
} 

同样只是回调instance_对应方法。

ModuleRegistry.cpp

储存NativeModule.cpp列表modules_,用来回调java端的NativeModule方法。

folly::Optional ModuleRegistry::getConfig(const std::string& name) {
        ``````
}

主要是得到NativeModule的方法名列表和常量列表,传给js端NativeModules.js中。

void ModuleRegistry::callNativeMethod(unsigned int moduleId, 
         unsigned int methodId, folly::dynamic&& params, int callId) {
  if (moduleId >= modules_.size()) {
    throw std::runtime_error(
      folly::to("moduleId ", moduleId,
         " out of range [0..", modules_.size(), ")"));
  }
  modules_[moduleId]->invoke(methodId, std::move(params), callId);
}

异步NativeModule方法的调用。调用JavaModuleWrapper.cpp的invoke方法,会调用JavaModuleWrapper.java的invoke方法。

MethodCallResult ModuleRegistry::callSerializableNativeHook(unsigned int moduleId,
        unsigned int methodId, folly::dynamic&& params) {
  if (moduleId >= modules_.size()) {
    throw std::runtime_error(
      folly::to("moduleId ", moduleId,
        "out of range [0..", modules_.size(), ")"));
  }
  return modules_[moduleId]->
       callSerializableNativeHook(methodId, std::move(params));
}

同步NativeModule方法的调用。调用JavaModuleWrapper.cpp的callSerializableNativeHook方法,这个方法会直接调用NativeModule方法,得到结果值并返回。

JavaModuleWrapper.cpp

void JavaNativeModule::invoke(unsigned int reactMethodId, 
         folly::dynamic&& params, int callId) {
  messageQueueThread_->runOnQueue(
         [this, reactMethodId, params=std::move(params), callId] {
    static auto invokeMethod = wrapper_->getClass()->getMethod
               ("invoke");
    #ifdef WITH_FBSYSTRACE
    if (callId != -1) {
      fbsystrace_end_async_flow(TRACE_TAG_REACT_APPS, "native", callId);
    }
    #endif
    invokeMethod(
      wrapper_,
      static_cast(reactMethodId),
      ReadableNativeArray::newObjectCxxArgs(std::move(params)).get());
  });
}

这个方法主要作用调用JavaModuleWrapper.java的invoke方法。wrapper_就是JavaModuleWrapper.java对应的实例。

MethodCallResult JavaNativeModule::callSerializableNativeHook(
           unsigned int reactMethodId, folly::dynamic&& params) {
  // TODO: evaluate whether calling through invoke is potentially faster
  if (reactMethodId >= syncMethods_.size()) {
    throw std::invalid_argument(
      folly::to("methodId ", reactMethodId,
           " out of range [0..", syncMethods_.size(), "]"));
  }

  auto& method = syncMethods_[reactMethodId];
  CHECK(method.hasValue() && method->isSyncHook()) 
         << "Trying to invoke a asynchronous method as synchronous hook";
  return method->invoke(instance_, wrapper_->getModule(), params);
}

直接调用NativeModule的方法,并将结果值返回。

Instance.cpp

主要作用是处理CatalystInstanceImpl对应的方法,还会创建一个NativeToJsBridge实例。

void Instance::initializeBridge(
    std::unique_ptr callback,
    std::shared_ptr jsef,
    std::shared_ptr jsQueue,
    std::shared_ptr moduleRegistry) {
  callback_ = std::move(callback);
  moduleRegistry_ = std::move(moduleRegistry);

  jsQueue->runOnQueueSync([this, &jsef, jsQueue]() mutable {
    nativeToJsBridge_ = folly::make_unique(
        jsef.get(), moduleRegistry_, jsQueue, callback_);

    std::lock_guard lock(m_syncMutex);
    m_syncReady = true;
    m_syncCV.notify_all();
  });

  CHECK(nativeToJsBridge_);
}

进行属性的赋值,并创建NativeToJsBridge实例。callback_是java端ReactCallback的实例,jsef用来得到生成JSCExecutor.cpp实例的工厂类。

void Instance::callJSFunction(std::string &&module, std::string &&method,
                              folly::dynamic &¶ms) {
  callback_->incrementPendingJSCalls();
  nativeToJsBridge_->callFunction(std::move(module), std::move(method),
                                  std::move(params));
}

先回调java端ReactCallback的incrementPendingJSCalls方法,然后调用nativeToJsBridge_的对应方法。

void Instance::callJSCallback(uint64_t callbackId, folly::dynamic &¶ms) {
  SystraceSection s("Instance::callJSCallback");
  callback_->incrementPendingJSCalls();
  nativeToJsBridge_->invokeCallback((double)callbackId, std::move(params));
}

先回调java端ReactCallback的incrementPendingJSCalls方法,然后调用nativeToJsBridge_的对应方法。

NativeToJsBridge.cpp

创建JsToNativeBridge实例和JSExecutor实例(即JSCExecutor.cpp实例),然后调用JSExecutor对应方法与js交互。
这个类还包含一个JsToNativeBridge类,用它来与java端交互的,即调用NativeModule,所有它拥有ModuleRegistry。

JsToNativeBridge

 JsToNativeBridge(std::shared_ptr registry,
                   std::shared_ptr callback)
    : m_registry(registry)
    , m_callback(callback) {}

得到ModuleRegistry属性。callback是java端的BridgeCallback实例。

void callNativeModules(
      JSExecutor& executor, folly::dynamic&& calls, bool isEndOfBatch) override {

    CHECK(m_registry || calls.empty()) <<
      "native module calls cannot be completed with no native modules";
    m_batchHadNativeModuleCalls = m_batchHadNativeModuleCalls || !calls.empty();

    for (auto& call : parseMethodCalls(std::move(calls))) {
      m_registry->callNativeMethod(call.moduleId, call.methodId, std::move(call.arguments), call.callId);
    }
    if (isEndOfBatch) {
      if (m_batchHadNativeModuleCalls) {
        m_callback->onBatchComplete();
        m_batchHadNativeModuleCalls = false;
      }
      m_callback->decrementPendingJSCalls();
    }
  }

我们可以看到通过m_registry的callNativeMethod方法,来回调java端对应的异步方法。parseMethodCalls是MethodCall.cpp中的方法,主要作用是将js端传递来的json格式数据转换成我们需要的数据格式。

 MethodCallResult callSerializableNativeHook(
      JSExecutor& executor, unsigned int moduleId, unsigned int methodId,
      folly::dynamic&& args) override {
    return m_registry->callSerializableNativeHook(moduleId, methodId, std::move(args));
  }

通过m_registry的callSerializableNativeHook,进行NativeModule同步方法的调用。

下面是NativeToJsBridge中的方法

NativeToJsBridge::NativeToJsBridge(
    JSExecutorFactory* jsExecutorFactory,
    std::shared_ptr registry,
    std::shared_ptr jsQueue,
    std::shared_ptr callback)
    : m_destroyed(std::make_shared(false))
    , m_delegate(std::make_shared(registry, callback))
    , m_executor(jsExecutorFactory->createJSExecutor(m_delegate, jsQueue))
    , m_executorMessageQueueThread(std::move(jsQueue)) {}

我们可以看到,创建了一个JsToNativeBridge实例m_delegate,以及JSExecutor实例m_executor(即JSCExecutor.cpp的实例)。

void NativeToJsBridge::callFunction(
    std::string&& module,
    std::string&& method,
    folly::dynamic&& arguments) {
  int systraceCookie = -1;
  #ifdef WITH_FBSYSTRACE
  systraceCookie = m_systraceCookie++;
  FbSystraceAsyncFlow::begin(
      TRACE_TAG_REACT_CXX_BRIDGE,
      "JSCall",
      systraceCookie);
  #endif

  runOnExecutorQueue([module = std::move(module), method = std::move(method), 
            arguments = std::move(arguments), systraceCookie]
    (JSExecutor* executor) {
      #ifdef WITH_FBSYSTRACE
      FbSystraceAsyncFlow::end(
          TRACE_TAG_REACT_CXX_BRIDGE,
          "JSCall",
          systraceCookie);
      SystraceSection s("NativeToJsBridge::callFunction", 
           "module", module, "method", method);
      #endif
    
      executor->callFunction(module, method, arguments);
    });
}

用JSExecutor的callFunction方法(JSCExecutor.cpp中有具体实现)。

void NativeToJsBridge::invokeCallback(double callbackId,
         folly::dynamic&& arguments) {
  int systraceCookie = -1;
  #ifdef WITH_FBSYSTRACE
  systraceCookie = m_systraceCookie++;
  FbSystraceAsyncFlow::begin(
      TRACE_TAG_REACT_CXX_BRIDGE,
      "",
      systraceCookie);
  #endif

  runOnExecutorQueue(
       [callbackId, arguments = std::move(arguments), systraceCookie]
    (JSExecutor* executor) {
      #ifdef WITH_FBSYSTRACE
      FbSystraceAsyncFlow::end(
          TRACE_TAG_REACT_CXX_BRIDGE,
          "",
          systraceCookie);
      SystraceSection s("NativeToJsBridge::invokeCallback");
      #endif
      executor->invokeCallback(callbackId, arguments);
    });
}

用JSExecutor的invokeCallback方法(JSCExecutor.cpp中有具体实现)。

JSCExecutor.cpp

真正js端进行交互了,直接调用js端的方法。

std::unique_ptr JSCExecutorFactory::createJSExecutor(
      std::shared_ptr delegate, 
      std::shared_ptr jsQueue) {
      return folly::make_unique(delegate, jsQueue, m_jscConfig);
    }

创建一个JSExecutor实例,持有一个JsToNativeBridge实例对象delegate,用来回调java端方法的。

void JSCExecutor::bindBridge() throw(JSException) {
      SystraceSection s("JSCExecutor::bindBridge");
      std::call_once(m_bindFlag, [this] {
        auto global = Object::getGlobalObject(m_context);
        auto batchedBridgeValue = global.getProperty("__fbBatchedBridge");
        if (batchedBridgeValue.isUndefined()) {
          auto requireBatchedBridge = 
               global.getProperty("__fbRequireBatchedBridge");
          if (!requireBatchedBridge.isUndefined()) {
            batchedBridgeValue = 
               requireBatchedBridge.asObject().callAsFunction({});
          }
          if (batchedBridgeValue.isUndefined()) {
            throw JSException("Could not get BatchedBridge,
               make sure your bundle is packaged correctly");
          }
        }

        auto batchedBridge = batchedBridgeValue.asObject();
        m_callFunctionReturnFlushedQueueJS = batchedBridge.getProperty("callFunctionReturnFlushedQueue").asObject();
        m_invokeCallbackAndReturnFlushedQueueJS = batchedBridge.getProperty("invokeCallbackAndReturnFlushedQueue").asObject();
        m_flushedQueueJS = batchedBridge.getProperty("flushedQueue").asObject();
        m_callFunctionReturnResultAndFlushedQueueJS = batchedBridge.getProperty("callFunctionReturnResultAndFlushedQueue").asObject();
      });
    }

这个方法搭建通信桥,拿到js端的实例对象和方法,用它们直接调用js端代码,'__fbBatchedBridge':是在BatchedBridge.js中定义的,其实是一个MessageQueue.js对象实例,得到四个方法的引用,通过它们直接调用js方法。

  1. m_callFunctionReturnFlushedQueueJS:调用js端JavaScriptModule对应方法,不接收调用的结果值。
  2. m_invokeCallbackAndReturnFlushedQueueJS:NativeModule异步方法得到结果值的回调。
  3. m_flushedQueueJS:得到js端发起的NativeModule异步方法请求列表。
  4. m_callFunctionReturnResultAndFlushedQueueJS:与m_callFunctionReturnFlushedQueueJS相同,但是能够接受js端调用的结果值。可惜这个方法没有被java端调用过。
void JSCExecutor::callFunction(const std::string& moduleId, 
       const std::string& methodId, const folly::dynamic& arguments) {
      SystraceSection s("JSCExecutor::callFunction");
      auto result = [&] {
        JSContextLock lock(m_context);
        try {
          if (!m_callFunctionReturnResultAndFlushedQueueJS) {
            bindBridge();
          }
          return m_callFunctionReturnFlushedQueueJS->callAsFunction({
            Value(m_context, String::createExpectingAscii(m_context, moduleId)),
            Value(m_context, String::createExpectingAscii(m_context, methodId)),
            Value::fromDynamic(m_context, std::move(arguments))
          });
        } catch (...) {
          std::throw_with_nested(
            std::runtime_error("Error calling " + moduleId + "." + methodId));
        }
      }();
      callNativeModules(std::move(result));
    }

这个方法会调用js端JavaScriptModule对应方法,并得到当前js端发起的NativeModule异步方法请求列表,然后调用callNativeModules方法,这个方法会调用JsToNativeBridge的callNativeModules方法,最终调用java端代码。

void JSCExecutor::invokeCallback(const double callbackId, 
       const folly::dynamic& arguments) {
      SystraceSection s("JSCExecutor::invokeCallback");
      auto result = [&] {
        JSContextLock lock(m_context);
        try {
          if (!m_invokeCallbackAndReturnFlushedQueueJS) {
            bindBridge();
          }
          return m_invokeCallbackAndReturnFlushedQueueJS->callAsFunction({
            Value::makeNumber(m_context, callbackId),
            Value::fromDynamic(m_context, std::move(arguments))
          });
        } catch (...) {
          std::throw_with_nested(
          std::runtime_error(folly::to("Error invoking callback ", 
                   callbackId)));
        }
      }();
      callNativeModules(std::move(result));
    }

与上个方法流程相同,调用js端对应方法,得到当前js端发起的NativeModule异步方法请求列表,再调用callNativeModules方法。

void JSCExecutor::callNativeModules(Value&& value) {
      SystraceSection s("JSCExecutor::callNativeModules");
      CHECK(m_delegate) << "Attempting to use native modules without a delegate";
      try {
        auto calls = value.toJSONString();
        m_delegate->callNativeModules(*this, folly::parseJson(calls), true);
      } catch (...) {
        std::string message = "Error in callNativeModules()";
        try {
          message += ":" + value.toString().str();
        } catch (...) {
          // ignored
        }
        std::throw_with_nested(std::runtime_error(message));
      }
    }

会调用JsToNativeBridge的callNativeModules方法。

JSValueRef JSCExecutor::nativeCallSyncHook(
          size_t argumentCount,
         const JSValueRef arguments[]) {
      if (argumentCount != 3) {
        throw std::invalid_argument("Got wrong number of args");
      }

      unsigned int moduleId = Value(m_context, arguments[0]).asUnsignedInteger();
      unsigned int methodId = Value(m_context, arguments[1]).asUnsignedInteger();
      folly::dynamic args = 
           folly::parseJson(Value(m_context, arguments[2]).toJSONString());

      if (!args.isArray()) {
        throw std::invalid_argument(
              folly::to(
               "method parameters should be array, but are ", args.typeName()));
      }

      MethodCallResult result = m_delegate->callSerializableNativeHook(
                                                                  *this,
                                                                 moduleId,
                                                                 methodId,
                                                                 std::move(args));
      if (!result.hasValue()) {
        return Value::makeUndefined(m_context);
      }
      return Value::fromDynamic(m_context, result.value());
    }

这个方法是由js端NativeModules.js中调用,当js端调用NativeModule的一个同步方法时,它就会调用到这个方法。

由它调用JsToNativeBridge->callSerializableNativeHook方法,然后调用ModuleRegistry->callSerializableNativeHook方法,继续调用JavaModuleWrapper->callSerializableNativeHook方法, 这个方法直接调用java端NativeModule对应方法,并返回结果值(注意这里没有调用JavaModuleWrapper.java和JavaMethodWrapper.java的invoke方法,因为通过回调的方法向js端传递结果值,不能直接返回结果值)。

这里说一下java和c++代码的相互调用,js和c++代码的相互调用。我们都知道java可以通过jni来实现与c++端代码的相互调用的。那么js怎么实现的呢,通过全局变量global,c++通过global获取js端的实例和方法,也可以向global全局变量中注册c++本地的方法,然后js端就可以通过global来调用c++方法了。例如installNativeHook<&JSCExecutor::nativeCallSyncHook>("nativeCallSyncHook");就是向全局变量中注册nativeCallSyncHook这个本地方法。

总结

java端->js端:

从CatalystInstanceImpl.cpp开始,调用Instance.cpp对应方法,再调用NativeToJsBridge.cpp对应方法,再调用JSCExecutor.cpp对应方法,最终由JSCExecutor调用js端方法(因为JSCExecutor持有js对象方法的实例)

js端->java端:

  1. 异步方法的调用流程:通过JSCExecutor.cpp的callNativeModules方法,回调JsToNativeBridge(在NativeToJsBridge.cpp中)的callNativeModules方法,回调ModuleRegistry.cpp的callNativeMethod方法,再调用JavaModuleWrapper.cpp的invoke方法,然后就调用到JavaModuleWrapper.java中的invoke方法。
  2. 同步方法的调用流程:通过JSCExecutor.cpp的nativeCallSyncHook方法,回调JsToNativeBridge(在NativeToJsBridge.cpp中)的callSerializableNativeHook方法,回调ModuleRegistry.cpp的callSerializableNativeHook方法,回调JavaModuleWrapper.cpp的callSerializableNativeHook方法,然后直接调用java端NativeModule对应的方法,返回结果值。

下节预览

这一章分析了c++端通信流程的源码,那么下一章分析js端的通信机制的源码。

你可能感兴趣的:(ReactNative 通信机制_c++端源码分析)