JsBridge

一、产生背景:

APP面临的挑战:

1.支持复杂业务:App 的业务越来越复杂,不仅仅是内部业务,还包含了大量外部的合作伙伴。如果采用传统的 App 开发方式很难应对日趋复杂的业务场景。

2.满足业务快速迭代的需求:App 开发一个不容忽视的问题,就是应用商店审核。由于审核的存在,App 上开发的业务会有一个统一排期,比如说月底会有新版本,那么所有的业务进度都得考虑 App 的排期计划。

3.性能:如果目前支付宝的业务都是用 Native 开发,大家看到的包大小得增加好几倍,这进一步增加了用户的升级成本。

二、核心:

双向通信的通道

JS 向 Native 发送消息: 调用相关功能、通知 Native 当前 JS 的相关状态等。

Native 向 JS 发送消息: 回溯调用结果、消息推送、通知 JS 当前 Native 的状态等。

三、提供哪些能力

3.1一般JSBridge中实现的通用功能

自定义titleBar

自定义titleBar上左右两侧按钮的功能及样式

打开一个新的webview来承接跳转的url

关闭自身webview

关闭前n个webview

监听resume、pause事件

下拉刷新

app唤起

3.2JSBridge中实现的业务功能

页面分享(微信、微博分享)

登录SDK页面呼启

支付功能

调用相机、图片上传等

定位信息获取

3.3 无法满足

1.不适合传输大量数据(大量数据建议用接口方式获取)

四:历史通信方式

Android端

Native调JS

只要遵循:”javascript: 方法名(‘参数,需要转为字符串’)”的规则即可

在4.4之前,调用的方式:

mWebView = new WebView(this);

mWebView.loadUrl("javascript: 方法名('参数,需要转为字符串')”);

4.4以后(包括4.4),使用以下方式:

mWebView.evaluateJavascript("javascript: 方法名('参数,需要转为字符串')”, 

    newValueCallback() {}

注:4.4之前Native通过loadUrl来调用JS方法,只能让某个JS方法执行,但是无法获取该方法的返回值,(通过js改变iframe.src把结果返回,这样执行效率较低。)4.4之后,通过evaluateJavascript异步调用JS方法,并且能在onReceiveValue中拿到返回值

JS调native

Js调用Native需要对WebView设置@JavascriptInterface注解

Native中通过addJavascriptInterface添加暴露出来的JS桥对象,然后再该对象内部声明对应的API方法。

注:在Android4.2以上(api17后),暴露的api要加上注解@JavascriptInterface,否则会找不到方法。

IOS端

native调js

Native调用js的方法比较简单,Native通过stringByEvaluatingJavaScriptFromString调用Html绑定在window上的函数。

Js调native

Native中通过引入官方提供的JavaScriptCore库(iOS7以上),然后可以将api绑定到JSContext上(然后Html中JS默认通过window.top.*可调用)。

注:iOS7才出现这种方式, 在这之前, js无法直接调用Native,  只能通过JSBridge方式间接调用,iOS原生本身是无法被JS调用的,但是通过引入官方提供的第三方”JavaScriptCore”,即可开放api给JS调用


五、JSbridge

基础:url scheme是一种类似于url的链接,是为了方便app直接互相调用设计的。具体来讲如果是系统的url scheme,则打开系统应用,否则找看是否有app注册这种scheme,打开对应app。

我们常说的 Deeplink 一般也是基于 URL Scheme 来实现的。一个 URI 的组成结构如下:

URI = scheme:[//authority]path[?query][#fragment]

// scheme    = http

// authority  = www.baidu.com

// path      = /link

// query      = url=xxxxx

authority = [userinfo@]host[:port]

注:这种scheme必须原生app注册后才会生效。

诞生原因:解决IOS和Android在和JS通信过程中的的低版本兼容问题

流程:H5->通过某种方式触发一个url->Native捕获到url,进行分析->原生做处理->Native调用H5的JSBridge对象传递回调。

方法:

Android 的Webview 提供了 shouldOverrideUrlLoading 方法来提供给 Native 拦截 H5 发送的 URL Scheme 
例如:

iOS 的 WKWebview 可以根据拦截到的 URL Scheme 和对应的参数执行相关的操作


六、实现JSBridge步骤分析:

第一步:设计出一个Native与JS交互的全局桥对象

第二步:JS如何调用Native

第三步:Native如何得知api被调用

第四步:分析url-参数和回调的格式

第五步:Native如何调用JS

第六步:H5中api方法的注册以及格式

一、首先有一个全局桥对象

varJSBridge =window.JSBridge || (window.JSBridge = {});

二、Js调用native

通过它的callHandler方法来调用原生的api,

判断是否有回调函数,如果有,生成一个回调函数id,并将id和对应回调添加进入回调函数集合responseCallbacks中。

通过特定的参数转换方法,将传入的数据,方法名一起,拼接成一个url scheme

//基本有用信息就是后面的callbackId,handlerName与data

//原生捕获到这个scheme后会进行分析

varuri = CUSTOM_PROTOCOL_SCHEME://API_Name:callbackId/handlerName?data

注:

正常来说是可以通过window.location.href达到发起网络请求的效果的,但是有一个很严重的问题,就是如果我们连续多次修改window.location.href的值,在Native层只能接收到最后一次请求,前面的请求都会被忽略掉。所以JS端发起网络请求的时候,需要使用iframe,这样就可以避免这个问题。

三、Native如何捕获scheme被触发

Android中(WebViewClient里),通过shouldoverrideurlloading可以捕获到url scheme的触发。

iOS中,UIWebView有个特性:在UIWebView内发起的所有网络请求,都可以通过delegate函数在Native层得到通知

四、URL参数和回调函数格式

1.根据api名,在本地找寻对应的api方法,并且记录该方法执行完后的回调函数id

2.根据提取出来的参数,根据定义好的参数进行转化

3.原生本地执行对应的api功能方法

4.功能执行完毕后,找到这次api调用对应的回调函数id,然后连同需要传递的参数信息,组装成一个JSON格式的参数

5.通过JSBridge通知H5页面回调

五、Native调用JS

到了这一步,就该Native通过JSBridge调用H5的JS方法或者通知H5进行回调了

Native通知H5页面进行回调:

数据格式为: Native通知H5回调的JSON格式。

Native主动调用H5方法:

Native主动调用H5方法时,数据格式是:{handlerName:api名,data:数据,callbackId:回调id}:

1.handlerName String型 需要调用的,h5中开放的api的名称

2.data JSON型 需要传递的数据,固定为JSON格式(因为我们固定H5中注册的方法接收的第一个参数必须是JSON,第二个是回调函数)

3.callbackId String型 原生生成的回调函数id,h5执行完毕后通过url scheme通知原生api成功执行,并传递参数

H5中API方法的注册以及格式

JSBridge.registerHandler('testH5Func',function(data,callback){

    alert('测试函数接收到数据:'+JSON.stringify(data));

    callback&&callback('测试回传数据…');

}

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

推荐阅读更多精彩内容