关于Android 和 h5 hybrid 开发的实践在网上有很多,Android自身就有一个webview,很多实践都是通过webview来实现的,以下是记录一下自己的实践。

1、Android与H5通讯方式


主要有两种:有基于url拦截的,也有基于prompt拦截的。


Android上面最火的这个开源项目 JsBridge就是基于url来进行拦截的



2、JsBridge原理和用法



原理



webview.loadUrl("javascript: alert('hello world')");



H5调用Android:webview可以拦截H5发起的任意url请求,webview通过约定的规则对拦截到的url进行处理(消费),即可实现H5调用Android


var ifm = document.createElement('iframe');
ifm.src = 'jsbridge://namespace.method?[...args]';




JsBridge上面已经封装了一层协议了。所以如果要自定义一套协议的话,可以参考下面这个方式

schema://classname/method?value={} // Android会收到这个url

schema即自行约定的私有协议,一般是公司名或者应用名


classname/method这两个一起,指定API名称,classname可以是Android的类名,当然也可以只是一个普通字符串(如namespace),只要Android方便识别即可


value是与method对应的参数


请求示例:


// H5直接指定包名和方法名,这样更容易扩展
iframe.src = `myschema://com.mycompany.hybrid.InteractHandler/toast?value={msg: 'hello world'}`

// 用约定的字符串也可以,Android拿到之后自己再做一次map
iframe.src = `myschema://interact/toast?value={msg: 'hello world'}



JsBridge用法1:

java注册handler方法以供h5里面的js调用

webView.registerHandler("submitFromWeb", new BridgeHandler() {
        @Override
        public void handler(String data, CallBackFunction function) {
            Log.i(TAG, "handler = submitFromWeb, data from web = " + data);
            function.onCallBack("submitFromWeb exe, response data from Java");
        }
    });



js可以通过下面的方式调用这个的java handler的“submitFromWeb”方法

WebViewJavascriptBridge.callHandler(
        'submitFromWeb'
        , {'param': str1}
        , function(responseData) {
            document.getElementById("show").innerHTML = "send get responseData from java, data = " + responseData
        }
    );



JsBridge用法2:

js注册handler方法以供Java里面的代码调用

WebViewJavascriptBridge.registerHandler("functionInJs", function(data, responseCallback) {
        document.getElementById("show").innerHTML = ("data from Java: = " + data);
        var responseData = "Javascript Says Right back aka!";
        responseCallback(responseData);
    });

java里面可以通过以下方式来调用这个js handler的“functionInJs"方法

webView.callHandler("functionInJs", new Gson().toJson(user), new CallBackFunction() {
        @Override
        public void onCallBack(String data) {

        }
    });



3、使用过程中的问题

1、最主要有一个是性能问题,通过iframe的src来进行跳转会遇到加载慢的问题,存在js已经加载完了,但是bridge还没有ready的情况,以至于有些情况下前端任务这个时间差太长了

2、是否存在jsbridge加载失败,可以通过cookiemanager传递一些内容?

3、用户信息的传递,通过token来传递

4、js中如何加上权限控制?java中如何控制js中代码的权限? 参考各种h5类型的开放平台实现


4、结语

性能问题可以考虑不采用url方式,而采用promt方式来实现?