jsbridge源码分析

本文分析的是https://github.com/lzyzsd/JsBridge这个开源库,

网络上很多关于这个库的分析,但是根据本人的理解思路,我也写了一份分析文章.

先讲一下该库的目的或者说功能:让js和java能够互相调用.

那么怎么互相调用呢?

我们以这个库提供的例子作为分析对象

java调用 js:

  webView.callHandler("functionInJs", "data from Java", new CallBackFunction() {

  @Override

  public void onCallBack(String data) {

      // TODO Auto-generated method stub

      Log.i(TAG, "reponse data from js " + data);

  }

});

对,就是这么简单,调用了webView.callHandler就可以了,注意,这个webView.callHandler并不是源生webview的方法,而是这个库实现的方法.

js调用java:

方法一:

window.WebViewJavascriptBridge.send(

    data

    , function(responseData) {

        document.getElementById("show").innerHTML = "repsonseData from java, data = " + responseData

    }

);

方法二:

window.WebViewJavascriptBridge.callHandler(

    'submitFromWeb'

    , {'param': '中文测试'}

    , function(responseData) {

        document.getElementById("show").innerHTML = "send get responseData from java, data = " + responseData

    }

);

这两种方法的的区别是方法一没有设置handlername.方法二有设置:    'submitFromWeb'

后面会讲设置这个handlername有什么作用.

我们以js调用java的方法一为入口,一步一步分析怎么调用到java的.

方法一最后调用到WebViewJavascriptBridge.js里:

//sendMessage add message, 触发native处理 sendMessage

function _doSend(message, responseCallback) {

    if (responseCallback) {

        var callbackId = 'cb_' + (uniqueId++) + '_' + new Date().getTime();

        responseCallbacks[callbackId] = responseCallback;

        message.callbackId = callbackId;

    }

    sendMessageQueue.push(message);

    messagingIframe.src = CUSTOM_PROTOCOL_SCHEME + '://' + QUEUE_HAS_MESSAGE;

}

这里的message就是前面的data.responseCallback就是前面的function(responseData),可以看到,这个 _doSend做了三件事:

一,生成一个callbackId.并把这个callbackId与responseCallback一起存到responseCallbacks

这个其实就是生成生个键值对,用于保存和识别传进来的responseCallback.以便后面使用.

二, 调用了 sendMessageQueue.push(message);

把消息放到sendMessageQueue里.

三,    messagingIframe.src = CUSTOM_PROTOCOL_SCHEME + '://' + QUEUE_HAS_MESSAGE;这个是什么呢?

这个其实就是用来触发java层的,通过修改iframe的src来触发webviewclient的shouldOverrideUrlLoading.这个src值是yy://__QUEUE_MESSAGE__/

此时我们切到com.github.lzyzsd.jsbridge.BridgeWebViewClient#shouldOverrideUrlLoading(android.webkit.WebView, android.webkit.WebResourceRequest):

调用到以下部分:

else if (url.startsWith(BridgeUtil.YY_OVERRIDE_SCHEMA)) { //

    webView.flushMessageQueue();

    return true;

可以看到,这里只调用了webView.flushMessageQueue();

这个方法只做了两件事:

一,调用loadUrl(jsUrl) 触发js调用,也就是说,我们刚从js到java代码,现在又回到js里了.这个jsUrl是BridgeUtil.JS_FETCH_QUEUE_FROM_JAVA,也就是

final static String JS_FETCH_QUEUE_FROM_JAVA = "javascript:WebViewJavascriptBridge._fetchQueue();";

二,创建CallBackFunction对象,并保存到responseCallbacks里.

我们看一下js里面做了什么:

// 提供给native调用,该函数作用:获取sendMessageQueue返回给native,由于android不能直接获取返回的内容,所以使用url shouldOverrideUrlLoading 的方式返回内容

function _fetchQueue() {

    var messageQueueString = JSON.stringify(sendMessageQueue);

    sendMessageQueue = [];

    //android can't read directly the return data, so we can reload iframe src to communicate with java

    messagingIframe.src = CUSTOM_PROTOCOL_SCHEME + '://return/_fetchQueue/' + encodeURIComponent(messageQueueString);

}

注释里已经写了很清楚了,还记得刚才我们讲js调到java的时候生成了一个message然后放到sendMessageQueue里的事吗.

这里就是把sendMessageQueue里的message取出来,转成json.然后拼装到messagingIframe.src 里.此时又会触发一次webviewclient的shouldOverrideUrlLoading:

if (url.startsWith(BridgeUtil.YY_RETURN_DATA)) { // 如果是返回数据

    webView.handlerReturnData(url);

    return true;

由于此次src值变了.此次走到的是上面这段.注意,此时我们又回到java代码了.我们看一下这个handlerReturnData的实现:

/**

    * 获取到CallBackFunction data执行调用并且从数据集移除

    * @param url

    */

void handlerReturnData(String url) {

  String functionName = BridgeUtil.getFunctionFromReturnUrl(url);

  CallBackFunction f = responseCallbacks.get(functionName);

  String data = BridgeUtil.getDataFromReturnUrl(url);

  if (f != null) {

      f.onCallBack(data);

      responseCallbacks.remove(functionName);

      return;

  }

}

这段代码就是取回第一次调用到shouldOverrideUrlLoading的时候我们设置的一个callback,也就是com.github.lzyzsd.jsbridge.BridgeWebView#flushMessageQueue里我们设置的new CallBackFunction()

  /**

    * 刷新消息队列

    */

void flushMessageQueue() {

  if (Thread.currentThread() == Looper.getMainLooper().getThread()) {

      loadUrl(BridgeUtil.JS_FETCH_QUEUE_FROM_JAVA, new CallBackFunction() {

        @Override

        public void onCallBack(String data) {

            // deserializeMessage 反序列化消息

            List list = null;

            try {

              list = Message.toArrayList(data);

            } catch (Exception e) {

                      e.printStackTrace();

              return;

            }

            if (list == null || list.size() == 0) {

              return;

            }

            for (int i = 0; i < list.size(); i++) {

              Message m = list.get(i);

              String responseId = m.getResponseId();

              // 是否是response  CallBackFunction

              if (!TextUtils.isEmpty(responseId)) {

                  CallBackFunction function = responseCallbacks.get(responseId);

                  String responseData = m.getResponseData();

                  function.onCallBack(responseData);

                  responseCallbacks.remove(responseId);

              } else {

                  CallBackFunction responseFunction = null;

                  // if had callbackId 如果有回调Id

                  final String callbackId = m.getCallbackId();

                  if (!TextUtils.isEmpty(callbackId)) {

                    responseFunction = new CallBackFunction() {

                        @Override

                        public void onCallBack(String data) {

                          Message responseMsg = new Message();

                          responseMsg.setResponseId(callbackId);

                          responseMsg.setResponseData(data);

                          queueMessage(responseMsg);

                        }

                    };

                  } else {

                    responseFunction = new CallBackFunction() {

                        @Override

                        public void onCallBack(String data) {

                          // do nothing

                        }

                    };

                  }

                  // BridgeHandler执行

                  BridgeHandler handler;

                  if (!TextUtils.isEmpty(m.getHandlerName())) {

                    handler = messageHandlers.get(m.getHandlerName());

                  } else {

                    handler = defaultHandler;

                  }

                  if (handler != null){

                    handler.handler(m.getData(), responseFunction);

                  }

              }

            }

        }

      });

  }

}

可以看出,最关键的调用都在这里了.我们重点一段一段贴出来分析一下:

String responseId = m.getResponseId();

// 是否是response  CallBackFunction

if (!TextUtils.isEmpty(responseId)) {

  CallBackFunction function = responseCallbacks.get(responseId);

  String responseData = m.getResponseData();

  function.onCallBack(responseData);

  responseCallbacks.remove(responseId);

}

先判断message是否有responseId.这个responseId是什么呢,这个先不分析,从js调用到java的流程里,我们没看到有设置这个.

final String callbackId = m.getCallbackId();

if (!TextUtils.isEmpty(callbackId)) {

  responseFunction = new CallBackFunction() {

      @Override

      public void onCallBack(String data) {

        Message responseMsg = new Message();

        responseMsg.setResponseId(callbackId);

        responseMsg.setResponseData(data);

        queueMessage(responseMsg);

      }

  };

callbackId这个我们设置了.在前面说的方法一里,

function(responseData) {

        document.getElementById("show").innerHTML = "repsonseData from java, data = " + responseData

    }

这个就是callbackId.所以这里创建了一个新的message.并且设置了setResponseId,然后调用了queueMessage.这个后面分析.

BridgeHandler handler;

if (!TextUtils.isEmpty(m.getHandlerName())) {

  handler = messageHandlers.get(m.getHandlerName());

} else {

  handler = defaultHandler;

}

if (handler != null){

  handler.handler(m.getData(), responseFunction);

}

这个messageHandlers又是什么呢.

我们在前面的方法一和方法二里有看到区别,其实方法一就是没有设置messageHandlers.方法二就是有设置messageHandlers.

这个messageHandlers通过com.github.lzyzsd.jsbridge.BridgeWebView#registerHandler进行注册.传入的参数是js和java双方约定好的.

/**

* register handler,so that javascript can call it

* 注册处理程序,以便javascript调用它

* @param handlerName handlerName

* @param handler BridgeHandler

*/

public void registerHandler(String handlerName, BridgeHandler handler) {

  if (handler != null) {

          // 添加至 Map

      messageHandlers.put(handlerName, handler);

  }

}

分析完后,我们就清楚了,假如是js调用到java.双方约定一个字符串作为handlerName,然后,js通过方法二调用到java.这个时候,java就会通过registerHandler注册的handler,来执行约定的事情.

我们来看一下刚才遗留的queueMessage,

这个方法最后走到

com.github.lzyzsd.jsbridge.BridgeWebView#dispatchMessage

  /**

    * 分发message 必须在主线程才分发成功

    * @param m Message

    */

void dispatchMessage(Message m) {

      String messageJson = m.toJson();

      //escape special characters for json string  为json字符串转义特殊字符

      messageJson = messageJson.replaceAll("(\\\\)([^utrn])", "\\\\\\\\$1$2");

      messageJson = messageJson.replaceAll("(?<=[^\\\\])(\")", "\\\\\"");

      String javascriptCommand = String.format(BridgeUtil.JS_HANDLE_MESSAGE_FROM_JAVA, messageJson);

      // 必须要找主线程才会将数据传递出去 --- 划重点

      if (Thread.currentThread() == Looper.getMainLooper().getThread()) {

          this.loadUrl(javascriptCommand);

      }

  }

也就是调用到了WebViewJavascriptBridge._handleMessageFromNative.这个在WebViewJavascriptBridge.js里.我们去看一下这个的实现:

//提供给native调用,receiveMessageQueue 在会在页面加载完后赋值为null,所以

function _handleMessageFromNative(messageJSON) {

    console.log(messageJSON);

    if (receiveMessageQueue) {

        receiveMessageQueue.push(messageJSON);

    }

    _dispatchMessageFromNative(messageJSON);


}

这里调用到    _dispatchMessageFromNative(messageJSON);

//提供给native使用,

function _dispatchMessageFromNative(messageJSON) {

    setTimeout(function() {

        var message = JSON.parse(messageJSON);

        var responseCallback;

        //java call finished, now need to call js callback function

        if (message.responseId) {

            responseCallback = responseCallbacks[message.responseId];

            if (!responseCallback) {

                return;

            }

            responseCallback(message.responseData);

            delete responseCallbacks[message.responseId];

        } else {

            //直接发送

            if (message.callbackId) {

                var callbackResponseId = message.callbackId;

                responseCallback = function(responseData) {

                    _doSend({

                        responseId: callbackResponseId,

                        responseData: responseData

                    });

                };

            }

            var handler = WebViewJavascriptBridge._messageHandler;

            if (message.handlerName) {

                handler = messageHandlers[message.handlerName];

            }

            //查找指定handler

            try {

                handler(message.data, responseCallback);

            } catch (exception) {

                if (typeof console != 'undefined') {

                    console.log("WebViewJavascriptBridge: WARNING: javascript handler threw.", message, exception);

                }

            }

        }

    });

}

这个的逻辑和前面的flushMessageQueue很像

if (message.responseId) {

    responseCallback = responseCallbacks[message.responseId];

    if (!responseCallback) {

        return;

    }

    responseCallback(message.responseData);

    delete responseCallbacks[message.responseId];

}

这里的responseCallback.实际上就是方法一中js调用java时,传入的 function(responseData),在前面分析_doSend的时候设置的.也就是调用了一次js调用java时传入的function(responseData).

至此,js调用java就结束了.

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
禁止转载,如需转载请通过简信或评论联系作者。
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,324评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,303评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,192评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,555评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,569评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,566评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,927评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,583评论 0 257
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,827评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,590评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,669评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,365评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,941评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,928评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,159评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,880评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,399评论 2 342

推荐阅读更多精彩内容