JSB 原理与实践

article/2025/10/25 19:23:31

 大厂技术  坚持周更  精选好文

什么是 JSB

我们开发的 h5 页面运行在端上的 WebView 容器之中,很多业务场景下 h5 需要依赖端上提供的信息/能力,这时我们需要一个可以连接原生运行环境和 JS 运行环境的桥梁 这个桥梁就是 JSB,JSB 让 Web 端和 Native 端得以实现双向通信

WebView 概述

WebView 是移动端中的一个控件,它为 JS 运行提供了一个沙箱环境。WebView 能够加载指定的 url,拦截页面发出的各种请求等各种页面控制功能,JSB 的实现就依赖于 WebView 暴露的各种接口。由于历史原因,安卓和 iOS 均有高低两套版本的 WebView 内核:

平台和版本WebView 内核
iOS 8+WKWebView
iOS 2-8UIWebView
Android 4.4+Chrome
Android 4.4-Webkit

PS: 下文中出现的高版本均代指 iOS 8+ 或 Android 4.4+,低版本则相反。

JSB 原理

要实现双向通信自然要依次实现 Native 向 Web 发送消息和 Web 向 Native 发送消息。

Native 向 Web 发送消息

Native 向 Web 发送消息基本原理上是在 WebView 容器中动态地执行一段 JS 脚本,通常情况下是调用一个挂载在全局上下文的方法。Android 和 iOS 均提供了不同的接口来实现这一过程。

方法

  • Android 高低版本存在两种直接执行 JS 字符串的方法:

Android 版本API特点
低版本WebView.loadUrl无法执行回调
高版本WebView.evaluateJavascript可以拿到 JS 执行完毕的返回值

  • iOS 高低版本同样存在两种不同的实现方式:

iOS 版本API特点
低版本UIWebView.stringByEvaluatingJavaScriptFromString无法执行回调
高版本WKWebView.evaluateJavaScript可以拿到 JS 执行完毕的返回值

实践

下面我们通过一个小 Demo 来看一下在 iOS 端实现 Native 向 Web 端发消息的实际效果:

本文所有 Demo 均运行在 iOS14.5 模拟器中,WebView 容器采用 WKWebView 内核

页面上半部分的 UI 是由 HTML + CSS 渲染所得,是一个纯静态的 webpage,中间的输入框和按钮是 Native 原生控件,直接覆盖在 WebView 容器之上。在 Native 按钮上绑定了一个点击事件:将文本框输入的字符视为 JS 字符串并调用相关 API 直接执行

可以看到当我们在文本框中输入下列字符并点击按钮后,h5 页面中 id 为 test 的 p 标签内容被修改了。

document.querySelector('#test').innerHTML = 'I am from native';

敏锐同学到这一步其实就已经知道我们在日常使用 JSB 时客户端是如何调用前端 JS 代码了,我们在刚刚的静态 html 文件中添加几行 JS 代码:

function evaluateByNative(params) {const p = document.createElement('p');p.innerText = params;document.body.appendChild(p);return 'Hello Bridge!';
}

在文本框中输入 evaluateByNative(23333),来看一下调用的结果:

可以看到 Native 端可以直接调用挂载在 window 上的全局方法并传入相应的函数执行参数并且在函数执行结束后 Native 端可以直接拿到执行成功的返回值。

Web 向 Native 发送消息

Web 向 Native 发送消息本质上就是某段 JS 代码的执行端上是可感知的,目前业界主流的实现方案有两种,分别是拦截式注入式

拦截式

和浏览器类似 WebView 中发出的所有请求都是可以被 Native 容器感知到的(是不是想到了Gecko),因此拦截式具体指的是 Native 拦截 Web 发出的 URL 请求,双方在此之前约定一个 JSB 请求格式,如果该请求是 JSB 则进行相应的处理,若不是则直接转发。

Native 拦截请求的钩子方法:

平台API
AndroidshouldOverrideUrlLoading
iOS 8+decidePolicyForNavigationAction
iOS 8-shouldStartLoadWithRequest

拦截式的基本流程如下

上述流程存在几个问题:

  1. 通过何种方式发出请求?

Web 端发出请求的方式非常多样,例如 <a/>iframe.srclocation.hrefajax 等,但 <a/> 需要用户手动触发,location.href 可能会导致页面跳转,安卓端拦截 ajax 的能力有所欠缺,因此绝大多数拦截式实现方案均采用iframe 来发送请求

  1. 如何规定 JSB 的请求格式?

一个标准的 URL 由 <scheme>://<host>:<port><path> 组成,相信大家都有过从微信或手机浏览器点击某个链接意外跳转到其他 App 的经历,如果有仔细留意过这些链接的 URL 你会发现目前主流 App 都有其专属的一个 scheme 来作为该应用的标识,例如微信的 URL scheme 就是 weixin://JSB 的实现借鉴这一思路,定制业务自身专属的一个 URL scheme 来作为 JSB 请求的标识,例如字节内部实现拦截式 JSB 的 SDK 中就定义了 bytedance:// 这样一个 scheme。

// Web 通过动态创建 iframe,将 src 设置为符合双端规范的 url scheme
const CUSTOM_PROTOCOL_SCHEME = 'prek'function web2Native(event) {    const messagingIframe = document.createElement('iframe');messagingIframe.style.display = 'none';messagingIframe.src = CUSTOM_PROTOCOL_SCHEME + '://' + event;document.documentElement.appendChild(messagingIframe);setTimeout(() => {document.documentElement.removeChild(messagingIframe);}, 200)
}

拦截式在双端都具有非常好的向下兼容性,曾经是最主流的 JSB 实现方案,但目前在高版本的系统中已经逐渐被淘汰,理由是它有如下几个劣势:

  • 连续发送时可能会造成消息丢失(可以使用消息队列解决该问题)

  • URL  字符串长度有限制

  • 性能一般,URL request 创建请求有一定的耗时(Android 端 200-400ms)

实践案例

同样用一个简单的 Demo2 来看一下如何使用拦截式实现 Web 向 Native 发送消息,这里实现了在 Web 端唤起 Native 的相册。

遵循上述实现方式,Web 发送消息的代码如下:

const CUSTOM_PROTOCOL_SCHEME = 'prek' // 自定义 url schemefunction web2Native(event_name) {const messagingIframe = document.createElement('iframe')messagingIframe.style.display = 'none'messagingIframe.src = CUSTOM_PROTOCOL_SCHEME + '://' + event_namedocument.documentElement.appendChild(messagingIframe)setTimeout(() => {document.documentElement.removeChild(messagingIframe)}, 0)
}const btn = document.querySelector('#btn')btn.onclick = () => {web2Native('openPhotoAlbum')
}

Native 侧通过 decidePolicyForNavigationAction 这一 delegate 实现请求拦截,解析 URL 参数,若 URL scheme 是 prek 则认为该请求是一个来自 Web 的 JSB 调用:

- (void)webView:(WKWebView *)webView decidePolicyForNavigationAction:(WKNavigationAction *)navigationAction decisionHandler:(void (^)(WKNavigationActionPolicy))decisionHandler {NSURL *url = navigationAction.request.URL;NSLog(@"拦截到 Web 发出的请求 = %@", url);if ([self isSchemeMatchPrek:url]) {NSString* host = url.host.lowercaseString;if ([host isEqualToString: @"openphotoalbum"]) {[self openCameraForWeb]; // 打开相册NSLog(@"打开相册");}decisionHandler(WKNavigationActionPolicyCancel);return;} else {decisionHandler(WKNavigationActionPolicyAllow);}
}

为了更清晰地看到 Native 拦截的结果,在上述代理方法中打个断点:

继续执行,Congratulation!模拟器的相册被打开了!

注入式

注入式的原理是通过 WebView 提供的接口向 JS 全局上下文对象(window)中注入对象或者方法,当 JS 调用时,可直接执行相应的 Native 代码逻辑,从而达到 Web 调用 Native 的目的。

Native 注入 API 的相关方法:

平台API特点
AndroidaddJavascriptInterface4.2 版本以下有安全风险
iOS 8+WKScriptMessageHandler
iOS 7+JavaSciptCore

JSContext *context = [webView valueForKeyPath:@"documentView.webView.mainFrame.javaScriptContext"];context[@"getAppInfo"] = ^(msg) {return @"ggl_2693";
};
window.getAppInfo(); // 'ggl_2693'

这种方法简单而直观,并且不存在参数长度限制和性能瓶颈等问题,目前主流的 JSB SDK 都将注入式方案作为优先使用的对象。注入式的实现非常简单,这里不做案例展示。

两种方案对比

为了更清晰地表达这两种方式的区别,这里贴一个对比表格:

方案兼容性性能参数长度限制
拦截式无兼容性问题较差,安卓端尤为明显有限制
注入式安卓4.2+ 和 iOS 7+以上可用较好

如何执行回调

通过上述介绍我们已经知道如何实现双端互相发送消息,但上述两个通信过程缺少了“回应”这一动作,原因就是上述步骤缺少了回调函数的执行。以拦截式为例,常见的一个 JSB 调用是 Web 获取当前 App 信息, Native 拦截到 bytedance://getAppInfo这样一个请求后将获取当前 App 信息,那获取完成后如何让 Web 端拿到该信息呢?

一个最简单的做法是类比 JSONP 的实现,我们可以在请求的 URL 上拼接回调方法的事件名,将该事件挂载在全局 window 上,由于 Native 端可以轻松执行 JS 代码,因此在完成端逻辑后直接执行该事件名对应的回调方法即可。以 getAppInfo 为例:

// Web
const uniqueID = 1 // 为防止事件名冲突,给每个 callback 设置一个唯一标识
function webCallNative(event, params, callback) {if (typeof callback === 'Function') {const callbackID = 'jsb_cb_' + (uniqueID++) + '_' + Date.now();window[callbackID] = callback}const params = {callback: callbackID}// 构造 url schemeconst src = 'bytedance://getAppInfo?' + JSON.stringify(params)...
}// Native
1. 解析传入的参数 'getAppInfo' 得知 Web 希望获取 AppInfo
2. 执行端逻辑获取 AppInfo
3. 执行参数中挂载在全局的 callback 方法,AppInfo 作为回调方法的参数

因此只要把相应的回调方法挂载在全局对象上,Native 即可把每次调用后的响应通过动态执行 JS 方法的形式传递到 Web 端,这样一来整个通信过程就实现了闭环。

串联双端通信的过程

现在我们已经知道如何实现两端互相发送消息以及执行回调了,但看起来并不好用:首先调用 JSB 时需要在方法名后拼接参数和对应的回调函数,其次回调函数还需要一个一个地挂载在全局对象上。

我们期望的使用方式其实是这样:

// Web
web.call('event1', {param1}, (res) => {...}) // 触发 native event1 执行
web.on('event2', (res) => {...})// Native 
// 这里用 js 代替,理解大致意思即可
native.call('event2', {param2}, (res) => {...}) // 触发 web event2 执行
native.on('event1', (res) => {...})

这里的 JSB 就像是一个跨越两端的 EventEmitter,因此需要 Web 和 Native 遵循同一套调度机制。

上图给出了 Web 调用 -> Native 监听的执行过程,同理 Native 调用 -> Web 监听也是同样的逻辑,只是把两边的实现调换一种语言,这里不赘述了。

贴一张其他同学画的时序图,帮助理解整个通信过程

Demo3 基于开源的 WebViewJavascriptBridge 演示了一套完整的通讯流程是怎样进行的,有兴趣的同学请自行戳源码地址 JSB_Demo 自行体验。(需要使用 Xcode 打开,会涉及一些客户端的知识,请配合文档和 Google 使用)。

一点感受

笔者所在业务使用的 bridge 即司内目前最新的 SDK,没有历史包袱、使用体验也非常良好。得益于客户端遵循该 SDK 配套的实现机制,即使完全不了解 JSB 原理的同学在与端上对接 bridge 时也几乎没有遇到障碍。倘若抛开公司完备的基础建设,想实现一个通用且好用的 JSB 并非易事,因此了解其中的门道还是非常有益的。(巨人的肩膀站久了,确实巴适得很????)

参考文献

深入浅出 JSBridge[4]

JSB 实战[5]

[1]

JSONP: https://en.wikipedia.org/wiki/JSONP

[2]

WebViewJavascriptBridge: https://github.com/marcuswestin/WebViewJavascriptBridge

[3]

JSB_Demo: https://code.byted.org/caocheng.viccc/JSB_Demo

[4]

深入浅出 JSBridge: https://juejin.cn/post/6936814903021797389#heading-8

[5]

JSB 实战: https://juejin.cn/post/6844903702721986568

❤️ 谢谢支持

以上便是本次分享的全部内容,希望对你有所帮助^_^

喜欢的话别忘了 分享、点赞、收藏 三连哦~。

欢迎关注公众号 ELab团队 收货大厂一手好文章~

我们来自字节跳动,是旗下大力教育前端部门,负责字节跳动教育全线产品前端开发工作。

我们围绕产品品质提升、开发效率、创意与前沿技术等方向沉淀与传播专业知识及案例,为业界贡献经验价值。包括但不限于性能监控、组件库、多端技术、Serverless、可视化搭建、音视频、人工智能、产品设计与营销等内容。

欢迎感兴趣的同学在评论区或使用内推码内推到作者部门拍砖哦 ????

字节跳动校/社招投递链接: https://job.toutiao.com/

内推码:7EZKXME


http://chatgpt.dhexx.cn/article/ZCsjdy6A.shtml

相关文章

html一个带图片的表格,Html表格

&#xfeff;&#xfeff; 在解说今天Html表单之前。还是先看张图片来刺激一下。 源代码超链接演示 仿百度搜索框 请输入要搜索的内容&#xff1a; 看了上图百度的搜索页面&#xff0c;有木有心动一下&#xff0c;别慌你也能够的。这就是我们今天要讲的表单。 什么是Html表单—…

最新百度云不限速软件

直接放图不废话。 下载链接&#xff1a;https://pan.baidu.com/s/1ZDo6xMjMW525sH7FUB6QEA提取码&#xff1a;xey9

获取百度云盘不限速下载软件 每秒达到10兆 这叫一个“爽”

公众号内回复&#xff1a;不限速 即可获取下载链接 下载资源啦 永远也不需要开通VIP 赶快去下载吧 小编亲自体验下载速度 行走在岁月的小巷&#xff0c;听风&#xff0c;读雨&#xff0c;夜色空寂&#xff0c;一切烟云&#xff0c;皆会慢慢散去&#xff0c;光阴眷顾&a…

一款百度网盘不限速下载工具

此工具收集于网络&#xff0c;如有侵权请联系删除&#xff01;&#xff01;&#xff01; 此工具仅用于个人学习&#xff0c;请勿用于商业获利&#xff0c;造成后果自负&#xff01;&#xff01;&#xff01; 这款免费的百度云高速下载工具&#xff1a; 界面美观&#xff0c;…

百度云下载不限速方法+软件

转自&#xff1a;https://www.52pojie.cn/thread-888047-1-1.html 1、Pandownload这个不用多说&#xff0c;功能强大。目前是我的主力。 2、速盘速盘用得比较少&#xff0c;大致和pandownload差不多。 3、百度云下载神器这种就我个人来说比较鸡肋&#xff0c;需要在网页版的百度…

无封号风险,2020最新百度网盘不限速下载软件,下载速度10M/S

前不久给大家带来的卢本伟大神修改的pandownload不知道大家还在没在用&#xff0c;能不能用&#xff0c; 今天小七就在给大家带来一款度盘高速下载工具&#xff0c;软件完全免费并且没有任何下载限制&#xff0c;而且相比pandownload需要登录百度账号&#xff0c;这款软件无需…

Linux下载工具photon,不限速、免配置的 Aria2 免费开源下载软件 Photon,替代迅雷的...

原标题&#xff1a;不限速、免配置的 Aria2 免费开源下载软件 Photon&#xff0c;替代迅雷的 谈到下载软件除迅雷、IDM 之外&#xff0c;想必很多人都听过 Aria2 的大名&#xff0c;它绝对是跨平台不限速的“神器级”下载工具&#xff0c;可由于它是「命令行」软件&#xff0c;…

百度云下载不限速方式集合

因为百度云限速严重&#xff0c;因此很多时候都是在寻找相应的工具去破解百度云&#xff0c;这里整理一下自己一般用的一些工具的方法。&#xff08;有些方式有些时候会被官方和谐&#xff0c;因此多试几种&#xff09; 文章目录 1、使用第三方工具pandownload2、使用油猴脚本配…

解决微云下载限速问题

1.首先将文件存到微云 2.打开手机我的设备&#xff0c;我的电脑&#xff08;也可以是好友&#xff09;&#xff0c;找到发送文件 3.点击微云&#xff0c;找到微云里面的文件&#xff0c;点击发送 4.就可以在电脑端不限速下载了 注意但是好像根据很多人的反馈4GB以上的好像发不过…

【更新】网盘不限速下载 2021.01.13

哈喽大家好嘞&#xff01; 最近一直都有好多朋友们反馈&#xff0c; 以往的百度网盘都不能用了&#xff0c; 安排&#xff0c;这就安排。 大家在学习知识的同时&#xff0c;不要忘记点赞呦&#xff01; ★ KinhDown 搜集了全网&#xff0c;可算是找到了一个能用的&#xf…

android http下载限速,安卓手机端两种让网盘不限速下载方法介绍

百度网盘已然成为分享型网盘中一家独大的“大佬”了。时代就是这样不管你喜不喜欢&#xff0c;上网总会遇到些百度网盘共享的文件需要下载。然而&#xff0c;百度网盘对免费用户的限速已经到了“感人”的地步了&#xff0c;常常十多KB/秒的速度真能让人崩溃&#xff01;虽说会员…

百度云盘不限速下载工具(附带开源源码)

百度下载资源限速? 进来看完摆脱束博&#xff01; 软件名称&#xff1a;antdownload 软件大小&#xff1a;21.4M 软件介绍&#xff1a; antdownload中文名蚂蚁下载器&#xff0c;是由Recursion开发的百度文件无限速批量下载器。软件支持解析百度用户分享网盘链接&#xff…

百度网盘下载不限速方法

百度网盘下载不限速方法&#xff1a; -------20230401-------- 使用软件&#xff1a;爱奇艺万能联播 众所周知&#xff0c;爱奇艺是百度旗下的一款产品&#xff0c;所以用爱奇艺万能联播的方法来实现下载百度网盘内容&#xff0c;本质上并没有破解百度网盘&#xff0c;所以没…

百度网盘如何不限速下载资源?

百度网盘破解限速小技巧&#xff0c;软件&#xff1a; Proxyee Down 这个软件能够安全有效高速的下载网盘资源,注意&#xff0c;使用前需要安装jdk&#xff0c;即配置好Java运行环境 https://github.com/proxyee-down-org/proxyee-down/wiki/%E8%BD%AF%E4%BB%B6%E4%B8%8B%E8…

pdproxy度盘下载器不限速(xp版也可用) v2021

简介&#xff1a; pdproxy度盘下载器不限速可以帮助用户快速下载网盘中的资源&#xff0c;同时还为用户提供解决第三方工具无法登陆百度账号的功能&#xff0c;功能十分强大&#xff0c;下载速度很快&#xff0c;突破了网盘的速度限制&#xff0c;并且免费提供给用户使用&…

最新百度网盘不限速下载软件,TuTDown,亲测,如图

百度网盘&#xff0c;想必大家都用过吧。它的龟速下载&#xff0c;是大伙吐糟最多的。也是大家最深恶痛绝网盘的一点 提起网速又不得不说起百度网盘&#xff0c;百度网盘是当初许多网盘幸存者之一&#xff0c;百度网盘使用非常方便&#xff0c;但百度网盘也有一个缺点&#xff…

一款网盘搜索神器 + 某度网盘不限速下载软件

众所周知&#xff0c;大家都是爱好学习的优秀青年&#xff0c;经常需要去各大网盘倒腾一些学习资料&#xff0c;但是市面上网盘众多&#xff0c;导致大家不能在第一时间找到所需的学习资料&#xff0c;以至心灰意冷&#xff0c;失去了学习的动力&#xff0c;放弃美好人生。作为…

精品:AG 1.3.8 for mac 百度网盘不限速下载软件

下载&#xff1a;AG 1.3.8 chrome插件 安装方法&#xff1a; 第一步&#xff1a;下载AG 1.3.8 chrome插件中的chrome.dmg和AG1.3.8安装包&#xff1a; ①打开AG 1.3.8.dmg完成AG的安装&#xff08;拖动到Applications文件夹进行安装&#xff09; ②解压chrome.dmg&#x…

不限速的免费下载神器,牛逼!

2020 年&#xff0c;某知名国产下载软件 X 雷发布了全新版本&#xff0c;改动颇大&#xff0c;但仍有部分网友对它感到失望。 比如花钱购入会员后&#xff0c;该限速依然限速&#xff0c;很多资源还不允许下载&#xff0c;等等。 “除了 X 雷外&#xff0c;市面上还有什么好用的…

分享这款不限速免费下载神器,能在任何操作系统上使用

应该不少网友在用迅雷这个下载器吧&#xff0c;虽然是功能强大的下载器&#xff0c;但自从它发布了全新版本&#xff0c;不少网友对它就有点失望。 那不用迅雷的话&#xff0c;肯定有小伙伴问&#xff0c;有没有更好用的下载软件来替代一下它呢&#xff1f; 所以今天&#xff…