DSBridge-Android 源码分析

一 Android WebView Js 原生API

Android WebView 提供了Js 和 WebView相互调用的接口,js 调用Android 代码通过

  1. @JavascriptInterface 注解
  2. WebView.addJavascriptInterface(Object object, String name) 方法

实现JS 和java 对象的映射。

同样 WebView 也提供了 java 调用Js 代码的机制。通过以下两个方法:

  1. WebView.evaluateJavascript(String script, ValueCallback<String> resultCallback);
  2. WebView.loadUrl(String script); Android 4.4 以下版本使用
private void evaluateJavascript(String script) {
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
        WebView.evaluateJavascript(String script, ValueCallback<String> resultCallback);
    } else {
        WebView.loadUrl(String script);
    }
}

二 DSBridge 分析

github 上提供了一个Js Bridage, DSBridge-Android, 分析下实现原理:

一共三个java 文件:

文件 功能
DWebView.java 继承WebView 封装了Js调用
CompletionHandler.java 处理异步请求使用
OnReturnValue.java 返回值 接口

DWebView 类 继承自 WebView 主要包括这几个函数

  1. init 在WebView 的构造函数中调动,完成一些WebView 的设置。
  2. injectJs
  3. evaluateJavascript(final String script);

1. init

init 函数中主要有两个部分处理一个是注册一个 WebChromeClient, 一个是调用addJavascriptInterface 接口注册一个 Js 调用Java的通用api

  1. super.setWebChromeClient(mWebChromeClient); 在mWebChromeClient 的回调中调用 injectJs 完成Js 注入
  2. super.addJavascriptInterface(new Object(){}, BRIDGE_NAME),这是DSBridge的核心功能,向 Js 页面注册一个通用的Js 对象,这个对象有一个 call 方法,通过这个call 方法实现对其它 Android Api 的调用,下面主要分析这个方法。

2. js 调用方式

在 js 页面调用 Andoid 代码时通过:dsBridge 为java evaluateJavascript 调用是的Object 在Js的映射对象,然后调用 这个对象的call 方法:

// init dsBridge
<script src="https://unpkg.com/dsbridge/dist/dsbridge.js"> </script>
var dsBridge=require("dsbridge")

//Call synchronously 
var str=dsBridge.call("testSyn", {msg: "testSyn"});

//Call asynchronously
dsBridge.call("testAsyn", {msg: "testAsyn"}, function (v) {
  alert(v);
})

3. Java 注册 js Api

java 代码注册,js 调用的函数都封装在 JsApi 这个对象中,注意是DWebView 的 setJavascriptInterface,不是原生WebView .

DWebView.setJavascriptInterface(new JsApi());

public class JsApi{

    @JavascriptInterface
    public void testAsyn(JSONObject jsonObject, CompletionHandler handler) throws JSONException {
        handler.complete(jsonObject.getString("msg")+" [ asyn call]");
    }
}

4. call 函数

call 需要使用 JavascriptInterface 注解注释,Js 中的三个参数被到这里被简化为两个参数,原因在Js 代码中分析。

  1. methodName java 方法名
  2. args 参数, 注意String 格式其实是Json 字符串

具体过程看代码注释:

@JavascriptInterface
public String call(String methodName, String args) {
        String error = "Js bridge method called, but there is " +
            "not a JavascriptInterface object, please set JavascriptInterface object first!";
       
        // 首先检查是否注册了Js api 相关的对象    
        if (jsb == null) {
            Log.e("SynWebView", error);
            return "";
        }

        // 获取注册的Js api 对象的Class 对象
        Class<?> cls = jsb.getClass();
        try {
            Method method;
            // 异步标记
            boolean asyn = false;    
            // String 类型的参数转化为  Json 对象
            JSONObject arg = new JSONObject(args);
            String callback = "";
            try {
                // 检查 json 对象中是否有_dscbstub 这个Key,如果有表示有回调Js的函数,是一个异步调用,
                // 然后移除,那么json对象中保存的都是参数
                // 有了对象,知道了对象的方法的String,通过反射获取这个方法。通过反射的参数可知
                // 方法的函数签名为:xxxMethod(JSONObject object, CompletionHandler handler);
                callback = arg.getString("_dscbstub");
                arg.remove("_dscbstub");
                method = cls.getDeclaredMethod(methodName,
                        new Class[]{JSONObject.class, CompletionHandler.class});
                asyn = true;
            } catch (Exception e) {
                method = cls.getDeclaredMethod(methodName, new Class[]{JSONObject.class});
            }

            // 错误检查
            if (method == null) {
                error = "ERROR! \n Not find method \"" + methodName + "\" implementation! ";
                Log.e("SynWebView", error);
                evaluateJavascript(String.format("alert(decodeURIComponent(\"%s\"})", error));
                return "";
            }

            // Js 调用的API 需要使用 @JavascriptInterface 注解,
            // 在4.4 以前的平台上有Js 安全漏洞,通过这个注解检查是否合法的API.
            // call 函数已经用 @JavascriptInterface 标注,是一个合法的API。jsp 对象由于绕过了WebView
            // 的 @JavascriptInterface 注解检查,需要手动校验。
            JavascriptInterface annotation = method.getAnnotation(JavascriptInterface.class);
            if (annotation != null) {
                Object ret;
                // 设置方法为可访问的
                method.setAccessible(true);
                if (asyn) {
                    // 异步调用, 讲异步调用的逻辑封装在CompletionHandler 中,
                    // 使用闭包的方式实现callback.
                    final String cb = callback;
                    ret = method.invoke(jsb, arg, new CompletionHandler() {
                        、、、

                        //  可以再method 方法中调用这个函数,实现异步。
                        private void complete(String retValue,boolean complete) {
                            try {
                                // retValue 为 method  执行的结果,complete 可以控制多次回调。
                                // 将callback 和参数组合为 javascript 语句,然后通过evaluateJavascript 
                                // 方法调用js 执行
                                if (retValue == null) retValue = "";
                                retValue = URLEncoder.encode(retValue, "UTF-8").replaceAll("\\+", "%20");
                                String script = String.format("%s(decodeURIComponent(\"%s\"));", cb, retValue);
                                // 将callback 方法从Html 的window 对象删除,原因在js 代码分析
                                if(complete) {
                                    script += "delete window."+cb;
                                }
                                evaluateJavascript(script);
                            } catch (UnsupportedEncodingException e) {
                                e.printStackTrace();
                            }
                        }
                    });
                } else {
                    // 同步调用
                    ret = method.invoke(jsb, arg);
                }
                if (ret == null) {
                    ret = "";
                }
                
                // 返回结果
                return ret.toString();
            } else {
                error = "Method " + methodName + " is not invoked, since  " +
                    "it is not declared with JavascriptInterface annotation! ";
                evaluateJavascript(String.format("alert('ERROR \\n%s')", error));
                Log.e("SynWebView", error);
            }
        } catch (Exception e) {
            evaluateJavascript(String.format("alert('ERROR! \\nCall failed:Function does not exist or parameter is invalid[%s]')", e.getMessage()));
            e.printStackTrace();
        }
        return "";
    }
    
    @Keep
    @JavascriptInterface
    public void returnValue(int id, String value) {
        OnReturnValue handler = handlerMap.get(id);
        if (handler != null) {
            handler.onValue(value);
            handlerMap.remove(id);
        }
    }
}, BRIDGE_NAME);            

5. injectJs 注入js 代码

在WebChromeClient 的回调中,会调用injectJs 方法注入js,onProgressChanged onReceivedTitle 保证在js 代码运行前 js注入完成

private WebChromeClient mWebChromeClient = new WebChromeClient() {

        @Override
        public void onProgressChanged(WebView view, int newProgress) {
            injectJs();
            
        }

        @Override
        public void onReceivedTitle(WebView view, String title) {
            injectJs();
        }
}

    private void injectJs() {
        evaluateJavascript("function getJsBridge(){window._dsf=window._dsf||{};return{call:function(b,a,c){\"function\"==typeof a&&(c=a,a={});if(\"function\"==typeof c){window.dscb=window.dscb||0;var d=\"dscb\"+window.dscb++;window[d]=c;a._dscbstub=d}a=JSON.stringify(a||{});return window._dswk?prompt(window._dswk+b,a):\"function\"==typeof _dsbridge?_dsbridge(b,a):_dsbridge.call(b,a)},register:function(b,a){\"object\"==typeof b?Object.assign(window._dsf,b):window._dsf[b]=a}}}dsBridge=getJsBridge();");
    }

6. javascript 代码注入分析

injectJs 调用的Js 代码如下:

function getJsBridge() {
    // window 对象的 _dsf 赋值, 如果没有定义过, 则定义为 {};
    // dsf 域用来保存 java 调用js 的function.
    window._dsf = window._dsf || {};
    
    // 返回 json 一个匿名json对象, json 对象包含两个function:call 和 register。
    return {
        // call function 包含三个参数, 方法名, 参数,回调函数,
        call: function (method, args, cb) {
            var ret = "";
            // 检查第二个参数类型是否为 function , 如果为function 则表示为回调函数
            if (typeof args == "function") {
                cb = args;
                args = {}
            }
            
            // 这一步处理很有技巧,在设置回调函数的时候,回调函数可能为匿名函数,
            // 在这里通过window 对象的一个域保存,避免垃圾回收和Java 回调的时候能够找到。
            // args 对象中 回调函数的Key 被设置为"_dscbstub", java 中是根据这个名字找到的callback
            // 也解释了java  中的call API 为两个参数, Js 中为三个参数的原因。
            if (typeof cb == "function") {
                window.dscb = window.dscb || 0;
                var cbName = "dscb" + window.dscb++;
                window[cbName] = cb;
                args["_dscbstub"] = cbName
            }
            args = JSON.stringify(args || {});
            if (window._dswk) {
                // debug 分支, window 的  _dswk 域决定
                ret = prompt(window._dswk + method, args)
            } else {
                // _dsbridge 对象为java 中调
                addJavascriptInterface(new Object(){}, BRIDGE_NAME) 映射的JS 对象
                if (typeof _dsbridge == "function") {
                    ret = _dsbridge(method, args)
                } else {
                    // 我们的代码走这里
                    ret = _dsbridge.call(method, args)
                }
            }
            return ret
        }, register: function (name, fun) {
            if (typeof name == "object") {
                Object.assign(window._dsf, name)
            } else {
                window._dsf[name] = fun
            }
        }
    }
}

// 最后把这个匿名对象挂在 window dsBridage 域下。
window.dsBridge = getJsBridge();

7 evaluateJavascript

DWebView 对 evaluateJavascript 做了两次封装,主要解决两个问题:

  1. 在非主线程中调用的问题, 通过handler post 到主线程中处理。
  2. 4.4 以前的版本兼容的问题。
    public void evaluateJavascript(final String script) {
        if (Looper.getMainLooper() == Looper.myLooper()) {
            _evaluateJavascript(script);
        } else {
            Message msg=new Message();
            msg.what=EXEC_SCRIPT;
            msg.obj=script;
            mainThreadHandler.sendMessage(msg);
        }
    }
    
    private void _evaluateJavascript(String script) {
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
            DWebView.super.evaluateJavascript(script, null);
        } else {
            loadUrl("javascript:" + script);
        }
    }    

8 java 调用js

android 的原生方式中java调用js 的接口已经很完善了。DSBridge 使用callHandler。js function 在调用前需要挂到
window._dsf 域下,参考 js代码的 register 函数。

//Register javascript function for Native invocation
 dsBridge.register('addValue',function(l,r){
     return l+r;
 })

java 中调用前指明window._dsf 下的function。 对代码做了一个约束。

DWebView.callHandler("addValue",new Object[]{1,"hello"},new OnReturnValue(){
       @Override
       public void onValue(String retValue) {
          Log.d("jsbridge","call succeed,return value is "+retValue);
       }
})


public void callHandler(String method, Object[] args, final OnReturnValue handler) {
        if (args == null) args = new Object[0];
        String arg = new JSONArray(Arrays.asList(args)).toString();
        String script = String.format("(window._dsf.%s||window.%s).apply(window._dsf||window,%s)", method,method, arg);
        if(handler!=null){
            script = String.format("%s.returnValue(%d,%s)",BRIDGE_NAME,callID, script);
            handlerMap.put(callID++, handler);
        }
        evaluateJavascript(script);
}
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,569评论 4 363
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,499评论 1 294
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 109,271评论 0 244
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,087评论 0 209
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,474评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,670评论 1 222
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,911评论 2 313
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,636评论 0 202
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,397评论 1 246
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,607评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,093评论 1 261
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,418评论 2 254
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,074评论 3 237
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,092评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,865评论 0 196
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,726评论 2 276
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,627评论 2 270

推荐阅读更多精彩内容