React Native 啟動流程詳細解析
導讀:本文以 react-native-cli 創(chuàng)建的示例工程(安卓部分)為例,分析 React Native 的啟動流程。
工程創(chuàng)建步驟可以參考官網(wǎng)。本文所分析 React Native 版本為 v0.64.2。
我們知道上述工程是一個安卓應用,打開 android/ 目錄下源碼文件,首先發(fā)現(xiàn)它創(chuàng)建了兩個 java 文件:MainApplication.java 和 MainActivity.java,分別做了應用以及主 Activity 的定義。
安卓應用的啟動流程是:在啟動第一個 activity 之前會創(chuàng)建一個全局唯一的 Application 對象。故在此我們先分析 MainApplication
MainApplication
public class MainApplication extends Application implements ReactApplication {
private final ReactNativeHost mReactNativeHost = new ReactNativeHost(this) {
@Override
public boolean getUseDeveloperSupport() {
return BuildConfig.DEBUG;
}
@Override
protected List<ReactPackage> getPackages() {
@SuppressWarnings("UnnecessaryLocalVariable")
List<ReactPackage> packages = new PackageList(this).getPackages();
// 其它對 packages 的操作
return packages;
}
@Override
protected String getJSMainModuleName() {
return "index";
}
}
@Override
public ReactNativeHost getReactNativeHost() {
return mReactNativeHost;
}
@Override
public void onCreate() {
super.onCreate();
SoLoader.init(this, /* native exopackage */ false);
initializeFlipper(this, getReactNativeHost().getReactInstanceManager());
}
MainApplication 繼承自 Application 類,并且實現(xiàn)了 ReactApplication 接口。在其中做的事情有:
1.創(chuàng)建成員變量 ReactNativeHost 的實例,并在創(chuàng)建過程中通過重寫 ReactNativeHost 類方法的方式,注入一些配置,包括:
- getUseDeveloperSupport: 配置是否開啟調(diào)試
- getPackages: 配置要加載的模塊
- getJSMainModuleName: 配置 js 模塊的入口文件名
2.在 onCreate 中:
- 調(diào)用 Soloader 庫。
Soloader是 facebook 推出的一個 so 文件加載庫,它能夠處理 so 文件的依賴在 react-native 中,所有框架相關的 so 文件都是通過SoLoader完成加載的 - 通過
ReactInstanceManager初始化 Flipper。Flipper是 facebook 推出的用于 debug ios、Android、React Native 應用的工具。
在這里簡要介紹下 ReactNativeHost 和 ReactInstanceManager
ReactNativeHost
ReactNativeHost 是個抽象類,開發(fā)者可以重寫其中的方法,其主要的作用是:在 application 中指定一些賦值操作,進而獲取 ReactInstanceManager 的實例。所以可以把 ReactNativeHost 作為將用戶自定義的參數(shù)賦值到 ReactInstanceManager 實例的中轉(zhuǎn)站。核心方法是: getReactInstanceManager,詳細分析見下文。
ReactInstanceManager
該類為核心類,主要負責管理 JS 的加載、維護生命周期、管理 JS 與 C++ 的交互等等??梢园?ReactInstanceManager 理解成 JS 與 C++ 的中轉(zhuǎn)橋梁。
MainActivity
接著看 MainActivity.java:
public class MainActivity extends ReactActivity {
@Override
protected String getMainComponentName() {
return "myProject";
}
}
MainActivity 類中僅重寫了 getMainComponentName 方法。該類繼承自 ReactActivity,我們再來看其 ReactActivity。
public abstract class ReactActivity extends AppCompatActivity
implements DefaultHardwareBackBtnHandler, PermissionAwareActivity {
private final ReactActivityDelegate mDelegate;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
mDelegate.onCreate(savedInstanceState);
}
ReactActivity 全權(quán)委托給 ReactActivityDelegate 來處理 onCreate 生命周期。來看 ReactActivityDelegate 的 onCreate。
protected void onCreate(Bundle savedInstanceState) {
String mainComponentName = getMainComponentName();
mReactDelegate =
new ReactDelegate(
getPlainActivity(), getReactNativeHost(), mainComponentName, getLaunchOptions()) {
@Override
protected ReactRootView createRootView() {
return ReactActivityDelegate.this.createRootView();
}
};
if (mMainComponentName != null) {
loadApp(mainComponentName);
}
}
這里首先創(chuàng)建了 ReactDelegate 實例。接著來看 loadApp 方法:
protected void loadApp(String appKey) {
mReactDelegate.loadApp(appKey);
getPlainActivity().setContentView(mReactDelegate.getReactRootView());
}
由此走到 ReactDelegate 實例的 loadApp 方法:
public void loadApp(String appKey) {
if (mReactRootView != null) {
throw new IllegalStateException("Cannot loadApp while app is already running.");
}
mReactRootView = createRootView();
mReactRootView.startReactApplication(
getReactNativeHost().getReactInstanceManager(), appKey, mLaunchOptions);
}
在這里做了三件事:創(chuàng)建 rootView (createRootView)、創(chuàng)建 ReactInstanceManager (getReactInstanceManager)、創(chuàng)建 ReactApplication (startReactApplication)。
createRootView
首先看下什么是 rootView。
public class ReactRootView extends FrameLayout implements RootView, ReactRoot { /* ... */}
ReactRootView 繼承自 FrameLayout,并且實現(xiàn)了 RootView、ReactRoot 兩個接口。FrameLayout 是安卓幾大布局中較為簡單的一個,整個界面被當成一塊空白備用區(qū)域,所有元素以左上角對齊堆疊。ReactRootView 繼承自 FrameLayout,表明其也是作為簡單布局而存在,UI 的繪制渲染都發(fā)生在上面。
getReactInstanceManager
ReactInstanceManager 是一個核心類,管理著 JS 的加載、C++ 和 JS 的交互、初始化參數(shù)等。最終調(diào)用來到 ReactNativeHost 類中的 createReactInstanceManager 方法:
protected ReactInstanceManager createReactInstanceManager() {
ReactInstanceManagerBuilder builder = /* ... */
for (ReactPackage reactPackage : getPackages()) {
builder.addPackage(reactPackage);
}
String jsBundleFile = getJSBundleFile();
if (jsBundleFile != null) {
builder.setJSBundleFile(jsBundleFile);
} else {
builder.setBundleAssetName(Assertions.assertNotNull(getBundleAssetName()));
}
ReactInstanceManager reactInstanceManager = builder.build();
return reactInstanceManager;
}
此處做的事情如下:
- 創(chuàng)建
ReactInstanceManagerBuilder實例。這里采用建造者模式來構(gòu)造ReactInstanceManager實例,故在此先傳入?yún)?shù)設定構(gòu)造者; - 把在
ReactNativeHost中注冊的packages都添加到ReactInstanceManagerBuilder實例中; - 如果
getJSBundleFile不為空,則加載對應的文件,否則加載默認的jsBundleFile; - 調(diào)用
builder.build方法。通過建造者真正構(gòu)造ReactInstanceManager實例
startReactApplication
public void startReactApplication(/* */) {
// ...
try {
// ...
mReactInstanceManager.createReactContextInBackground();
} finally {
// ...
}
}
最終執(zhí)行到 ReactInstanceManager 的 createReactContextInBackground 方法中。最后經(jīng)過調(diào)用鏈:recreateReactContextInBackgroundInner() -> recreateReactContextInBackgroundFromBundleLoader() -> recreateReactContextInBackground() -> runCreateReactContextOnNewThread()
runCreateReactContextOnNewThread 主要做了兩件事:
- 創(chuàng)建一個新的線程,并在新線程中通過
createReactContext創(chuàng)建ReactContext上下文; - 通過
setupReactContext來設置上下文環(huán)境,并最終調(diào)用到AppRegistry.js啟動App。
詳細分析我們放到另一篇文章:React Native startReactApplication 流程梳理。
總結(jié)
總結(jié)本文,通過 react-native-cli 創(chuàng)建的示例工程(安卓部分)為例,順著兩個類 MainApplication 和 MainActivity 的執(zhí)行流程,抓住主干邏輯,最終梳理出了 React Native 從開始啟動至執(zhí)行用戶 js 文件的過程??梢钥吹剑?/p>
MainApplication 的作用主要是傳入用戶的配置,并做 so 庫以及應用 debug 工具的初始化工作;
MainActivity 的作用主要是:
- 為應用創(chuàng)建
rootView布局容器; - 創(chuàng)建
ReactInstanceManager核心類,用于后續(xù)管理 JS 的加載、C++ 和 JS 的交互、初始化參數(shù)等; - 通過
startReactApplication來創(chuàng)建ReactContext上下文,并最終調(diào)用到AppRegistry.js啟動App。
到此這篇關于React Native 啟動流程簡析的文章就介紹到這了,更多相關React Native 啟動內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
ReactNative頁面跳轉(zhuǎn)Navigator實現(xiàn)的示例代碼
本篇文章主要介紹了ReactNative頁面跳轉(zhuǎn)Navigator實現(xiàn)的示例代碼,具有一定的參考價值,有興趣的可以了解一下2017-08-08
react antd如何防止一份數(shù)據(jù)多次提交
這篇文章主要介紹了react antd如何防止一份數(shù)據(jù)多次提交問題,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2023-10-10
React操作真實DOM實現(xiàn)動態(tài)吸底部的示例
本篇文章主要介紹了React操作真實DOM實現(xiàn)動態(tài)吸底部的示例,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2017-10-10
react中使用redux-persist做持久化儲存的過程記錄
這篇文章主要介紹了react中使用redux-persist做持久化儲存的相關資料,本文給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下2024-01-01
React父組件數(shù)據(jù)實時更新了,子組件沒有更新的問題
這篇文章主要介紹了React父組件數(shù)據(jù)實時更新了,子組件沒有更新的問題,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2024-03-03

