成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

React Native Debug原理淺析

avwu / 3340人閱讀

摘要:過(guò)程涉及到三個(gè)對(duì)象,一個(gè)是或,一個(gè)是,另外一個(gè)就是瀏覽器或或其他。在中進(jìn)行配置了,也就是會(huì)執(zhí)行腳本。然后會(huì)給這個(gè)注冊(cè)一些監(jiān)聽(tīng)在收到消息時(shí)會(huì)回調(diào)。發(fā)出一個(gè)消息讓瀏覽器準(zhǔn)備的運(yùn)行環(huán)境在收到消息會(huì)調(diào)用。

第一次在segmentfault寫(xiě)博客,很緊張~~~公司項(xiàng)目上ReactNative,之前也是沒(méi)有接觸過(guò),所以也是一邊學(xué)習(xí)一邊做項(xiàng)目了,最近騰出手來(lái)更新總結(jié)了一下RN的Debug的一個(gè)小知識(shí)點(diǎn),不是說(shuō)怎么去Debug,而是Debug的代碼原理,下面開(kāi)始正文。

Debug過(guò)程涉及到三個(gè)對(duì)象,一個(gè)是App(Android或iOS),一個(gè)是Server,另外一個(gè)就是瀏覽器(Chrome或FireFox或其他)。Server是App和瀏覽器之間通信的橋梁,比如App發(fā)Http請(qǐng)求給Server,Server再通過(guò)WebSocket發(fā)送給瀏覽器,反過(guò)來(lái)也是。首先肯定需要準(zhǔn)備一下中介,就是Server

1.Server

這里的Server不用專門準(zhǔn)備一臺(tái)服務(wù)器,只需要配置一個(gè)Node.js環(huán)境,然后啟動(dòng)npm start就行。npm start在package.json中進(jìn)行配置了,也就是會(huì)執(zhí)行cli.js腳本。

"scripts": {
    "start": "node node_modules/react-native/local-cli/cli.js start"
  },

然后cli.js會(huì)執(zhí)行runServer.js,在這里啟動(dòng)一個(gè)NodeJS Server:

const serverInstance = args.https
    ? https.createServer(
        {
          key: fs.readFileSync(args.key),
          cert: fs.readFileSync(args.cert),
        },
        app,
      )
    : http.createServer(app);

  serverInstance.listen(args.port, args.host, 511, function() {
    attachHMRServer({
      httpServer: serverInstance,
      path: "/hot",
      packagerServer,
    });

    wsProxy = webSocketProxy.attachToServer(serverInstance, "/debugger-proxy");
    ms = messageSocket.attachToServer(serverInstance, "/message");
    readyCallback(reporter);
  });

有了中介Server后就可以建立App與瀏覽器之間的關(guān)系了。

2.建立連接

在手機(jī)菜單中點(diǎn)擊Debug JS Remotely,App就會(huì)發(fā)出一個(gè)Http請(qǐng)求

GET /launch-js-devtools HTTP/1.1

Server接收到這個(gè)請(qǐng)求會(huì)執(zhí)行opn操作,主要做兩件事:

打開(kāi)Chrome的一個(gè)tab

讓這個(gè)tab打開(kāi)urlhttp://localhost:8081/debugger-ui/

這個(gè)界面就是我們打開(kāi)Debug時(shí)在瀏覽器見(jiàn)到的第一個(gè)界面

這個(gè)界面的文件就是Server的index.html,我截取了body的代碼:


  

React Native JS code runs as a web worker inside this tab.

Press ??I to open Developer Tools. Enable Pause On Caught Exceptions for a better debugging experience.

You may also install the standalone version of React Developer Tools to inspect the React component hierarchy, their props, and state.

Status: Loading...

瀏覽器在執(zhí)行index.html的時(shí)候會(huì)發(fā)出下面的請(qǐng)求:

GET /debugger-proxy?role=debugger&name=Chrome HTTP/1.1

我們來(lái)看看發(fā)出這個(gè)請(qǐng)求有什么目的,扒一扒源碼:

function connectToDebuggerProxy() {
    const ws = new WebSocket("ws://" + window.location.host + "/debugger-proxy?role=debugger&name=Chrome"); //Chrome通過(guò)websocket和Packager保持通訊
  
      //WebSocket注冊(cè)監(jiān)聽(tīng)
    ws.onopen = function() {
      Page.setState({status: {type: "connecting"}});
    };

    ws.onmessage = async function(message) {
      if (!message.data) {
        return;
      }
      const object = JSON.parse(message.data);

      if (object.$event === "client-disconnected") {
        shutdownJSRuntime();
        Page.setState({status: {type: "disconnected"}});
        return;
      }

      if (!object.method) {
        return;
      }

      // Special message that asks for a new JS runtime
      if (object.method === "prepareJSRuntime") {
        shutdownJSRuntime();
        console.clear();
        createJSRuntime();
        ws.send(JSON.stringify({replyID: object.id}));
        Page.setState({status: {type: "connected", id: object.id}});
      } else if (object.method === "$disconnected") {
        shutdownJSRuntime();
        Page.setState({status: {type: "disconnected"}});
      } else if (object.method === "executeApplicationScript") {
        worker.postMessage({
          ...object,
          url: await getBlobUrl(object.url),
        });
      } else {
        // Otherwise, pass through to the worker.
        worker.postMessage(object);
      }
    };

    ws.onclose = function(error) {
      shutdownJSRuntime();
      Page.setState({status: {type: "error", error}});
      if (error.reason) {
        console.warn(error.reason);
      }
      setTimeout(connectToDebuggerProxy, 500);
    };

    // Let debuggerWorker.js know when we"re not visible so that we can warn about
    // poor performance when using remote debugging.
    document.addEventListener("visibilitychange", updateVisibility, false);
  }

首先就是通過(guò)new WebSocket瀏覽器建立與Server的聯(lián)系,WebSocket就是可以保持長(zhǎng)連接的全雙工通信協(xié)議,在握手階段通過(guò)Http進(jìn)行,后面就和Http沒(méi)有什么關(guān)系了。然后會(huì)給這個(gè)webSocket注冊(cè)一些監(jiān)聽(tīng):

ws.onopen 
ws.onmessage
ws.onclose

在webSocket收到消息時(shí)會(huì)回調(diào)ws.onmessage。

到這里App和瀏覽器之間就已經(jīng)建立連接了,接下來(lái)App會(huì)發(fā)出幾個(gè)消息讓瀏覽器加載需要調(diào)試的代碼, 接著往下看。

3.加載調(diào)試代碼

首先需要強(qiáng)調(diào)的就是瀏覽器加載項(xiàng)目代碼肯定不能在UI線程加載吧,要不然肯定影響瀏覽器的正常工作。那怎么去加載?啟一個(gè)后臺(tái)線程,有的小伙伴就要不信了,別急,我們接著去扒一扒源碼。
App發(fā)出一個(gè)消息讓瀏覽器準(zhǔn)備JS的運(yùn)行環(huán)境:

在收到‘prepareJSRuntime’消息會(huì)調(diào)用createJSRuntime。
// Special message that asks for a new JS runtime
      if (object.method === "prepareJSRuntime") {
        shutdownJSRuntime();
        console.clear();
        createJSRuntime();
        ws.send(JSON.stringify({replyID: object.id}));
        Page.setState({status: {type: "connected", id: object.id}});
      } else if (object.method === "$disconnected") {
        shutdownJSRuntime();
        Page.setState({status: {type: "disconnected"}});
      } else if (object.method === "executeApplicationScript") {
        worker.postMessage({
          ...object,
          url: await getBlobUrl(object.url),
        });
      } else {
        // Otherwise, pass through to the worker.
        worker.postMessage(object);
      }

接著看‘createJSRuntime’這個(gè)函數(shù), 主要工作就是‘new Worker’,看下Worker的定義:

Web Workers is a simple means for web content to run scripts in 
background threads. The worker thread can perform tasks without
interfering with the user interface.
也就是會(huì)起一個(gè)后臺(tái)線程,來(lái)運(yùn)行‘debuggerWorker.js’這個(gè)腳本。
function createJSRuntime() {
      // This worker will run the application JavaScript code,
      // making sure that it"s run in an environment without a global
      // document, to make it consistent with the JSC executor environment.
      worker = new Worker("debuggerWorker.js");
      worker.onmessage = function(message) {
        ws.send(JSON.stringify(message.data));
      };
      window.onbeforeunload = function() {
        return "If you reload this page, it is going to break the debugging session. " +
          "You should press" + refreshShortcut + "in simulator to reload.";
      };
      updateVisibility();
}

接著看看debuggerWorker.js,主要就是一個(gè)消息的監(jiān)聽(tīng),可以看到在messageHandlers里主要處理兩類消息:

"executeApplicationScript", "setDebuggerVisibility"
/* global __fbBatchedBridge, self, importScripts, postMessage, onmessage: true */
/* eslint no-unused-vars: 0 */

"use strict";

onmessage = (function() {
  var visibilityState;
  var showVisibilityWarning = (function() {
    var hasWarned = false;
    return function() {
      // Wait until `YellowBox` gets initialized before displaying the warning.
      if (hasWarned || console.warn.toString().includes("[native code]")) {
        return;
      }
      hasWarned = true;
      console.warn(
        "Remote debugger is in a background tab which may cause apps to " +
        "perform slowly. Fix this by foregrounding the tab (or opening it in " +
        "a separate window)."
      );
    };
  })();

  var messageHandlers = {
    "executeApplicationScript": function(message, sendReply) {
      for (var key in message.inject) {
        self[key] = JSON.parse(message.inject[key]);
      }
      var error;
      try {
        importScripts(message.url);
      } catch (err) {
        error = err.message;
      }
      sendReply(null /* result */, error);
    },
    "setDebuggerVisibility": function(message) {
      visibilityState = message.visibilityState;
    },
  };

  return function(message) {
    if (visibilityState === "hidden") {
      showVisibilityWarning();
    }

    var object = message.data;

    var sendReply = function(result, error) {
      postMessage({replyID: object.id, result: result, error: error});
    };

    var handler = messageHandlers[object.method];
    if (handler) {
      // Special cased handlers
      handler(object, sendReply);
    } else {
      // Other methods get called on the bridge
      var returnValue = [[], [], [], 0];
      var error;
      try {
        if (typeof __fbBatchedBridge === "object") {
          returnValue = __fbBatchedBridge[object.method].apply(null, object.arguments);
        } else {
          error = "Failed to call function, __fbBatchedBridge is undefined";
        }
      } catch (err) {
        error = err.message;
      } finally {
        sendReply(JSON.stringify(returnValue), error);
      }
    }
  };
})();

App在點(diǎn)擊調(diào)試的時(shí)候會(huì)給瀏覽器還發(fā)送這么一個(gè)‘executeApplicationScript’消息,讓瀏覽器去加載項(xiàng)目代碼:

這個(gè)messageEvent的數(shù)據(jù)比較多,我就截取一部分,里面包含了方法名,url(這個(gè)url就是后面瀏覽器需要去下載bundle的地方),inject包含的數(shù)據(jù)最多,主要是會(huì)賦值給瀏覽器全局對(duì)象的方法。

{
  "id": 1,
  "method": "executeApplicationScript",
  "url": "http://localhost:8081/index.android.bundle?platform=android&dev=true&minify=false",
  "inject": {
    "__fbBatchedBridgeConfig": "{"remoteModuleConfig":[["AccessibilityInfo",{},["isTouchExplorationEnabled"]],["LocationObserver",{},["getCurrentPosition","startObserving","stopObserving"]],["CameraRollManager",{},["getPhotos","saveToCameraRoll"],[0,1]],["NetInfo",{},["getCurrentConnectivity","isConnectionMetered"],[0,1]],["PlatformConstants",{"ServerHost":"localhost:8081","reactNativeVersion":{"patch":0,"prerelease":null,"minor":51,"major":0},"Version":21,"isTesting":false}],["TimePickerAndroid",{}
}

webSocket首先接收到這個(gè)消息, 然后通過(guò)worker.postMessage給上面的worker發(fā)送‘executeApplicationScript’消息

ws.onmessage = async function(message) {
     ......
      // Special message that asks for a new JS runtime
      if (object.method === "prepareJSRuntime") {
        shutdownJSRuntime();
        console.clear();
        createJSRuntime();
        ws.send(JSON.stringify({replyID: object.id}));
        Page.setState({status: {type: "connected", id: object.id}});
      } else if (object.method === "$disconnected") {
        shutdownJSRuntime();
        Page.setState({status: {type: "disconnected"}});
      } else if (object.method === "executeApplicationScript") {
        worker.postMessage({
          ...object,
          url: await getBlobUrl(object.url),
        });
      } else {
        // Otherwise, pass through to the worker.
        worker.postMessage(object);
      }
    };

worker接收到這個(gè)消息在messageHandlers找到相應(yīng)的處理方法,在里面首選循環(huán)取出inject里面的字段和value然后賦值給self,在這里我理解就是這個(gè)worker線程的全局對(duì)象,然后通過(guò) importScripts(message.url)去加載bundle。

var messageHandlers = {
    "executeApplicationScript": function(message, sendReply) {
      for (var key in message.inject) {
        self[key] = JSON.parse(message.inject[key]);
      }
      var error;
      try {
        importScripts(message.url);
      } catch (err) {
        error = err.message;
      }
      sendReply(null /* result */, error);
    },
    ......
  };

為了證明我上面的分析沒(méi)錯(cuò),決定捉包看下發(fā)起的請(qǐng)求是不是這樣的:

在加載bundle后面還有一個(gè)map,體積也很大,有1.74MB的體積,這個(gè)是用于映射bundle里面的代碼成一個(gè)個(gè)工程項(xiàng)目里的類文件,這樣就和在代碼編譯器里面調(diào)試效果一樣了。

4.總結(jié)

根據(jù)上面的捉包請(qǐng)求簡(jiǎn)單總結(jié)下建立連接的過(guò)程,首先通過(guò)/launch-jsdevtools打開(kāi)調(diào)試Tab,瀏覽器通過(guò)/debugger-proxy建立與Server的WebSocket連接,然后瀏覽器打開(kāi)index.html文件,發(fā)起/debugger-ui/debuggerWorker.js建立后臺(tái)線程,通過(guò)這個(gè)后臺(tái)線程加載bundle。

到這里建立Debug連接的原理分析就差不多了,希望對(duì)小伙伴們有幫助,歡迎點(diǎn)贊和關(guān)注哈。

謝謝大家!

文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/51959.html

相關(guān)文章

  • React Native Debug原理淺析

    摘要:過(guò)程涉及到三個(gè)對(duì)象,一個(gè)是或,一個(gè)是,另外一個(gè)就是瀏覽器或或其他。在中進(jìn)行配置了,也就是會(huì)執(zhí)行腳本。然后會(huì)給這個(gè)注冊(cè)一些監(jiān)聽(tīng)在收到消息時(shí)會(huì)回調(diào)。發(fā)出一個(gè)消息讓瀏覽器準(zhǔn)備的運(yùn)行環(huán)境在收到消息會(huì)調(diào)用。 第一次在segmentfault寫(xiě)博客,很緊張~~~公司項(xiàng)目上ReactNative,之前也是沒(méi)有接觸過(guò),所以也是一邊學(xué)習(xí)一邊做項(xiàng)目了,最近騰出手來(lái)更新總結(jié)了一下RN的Debug的一個(gè)小知識(shí)...

    senntyou 評(píng)論0 收藏0
  • 淺析React之事件系統(tǒng)(二)

    摘要:因?yàn)樽柚故录芭莸男袨橹荒苡糜诤铣墒录?,沒(méi)法阻止原生事件的冒泡。同時(shí)的創(chuàng)建和冒泡是在原生事件冒泡到最頂層的之后的。淺析之事件系統(tǒng)一 上篇文章中,我們談到了React事件系統(tǒng)的實(shí)現(xiàn)方式,和在React中使用原生事件的方法,那么這篇文章我們來(lái)繼續(xù)分析下,看看React中合成事件和原生事件混用的各種情況。 上一個(gè)例子 在上篇文章中,我們舉了個(gè)例子。為了防止大家不記得,我們來(lái)看看那個(gè)例子的代...

    villainhr 評(píng)論0 收藏0
  • SegmentFault 技術(shù)周刊 Vol.4 - 這份 Android 有點(diǎn)甜

    摘要:閱讀本期周刊,你將快速入門,開(kāi)啟甜蜜之旅。然則的原理負(fù)責(zé)發(fā)送以及處理消息,創(chuàng)建消息隊(duì)列并不斷從隊(duì)列中取出消息交給,則用于保存消息。 showImg(/img/bVCN99?w=900&h=385); 2016 年 8 月,Android 7.0 Nougat(牛軋?zhí)牵┱桨l(fā)布,那么問(wèn)題來(lái)了,你 Marshmallow 了么(? -? ?) Cupcake、Donut、Gingerbre...

    jay_tian 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

最新活動(dòng)
閱讀需要支付1元查看
<