Hybrid App:狭义上是App内嵌WebView组件,再在WebView上使用页面的方案。广义上包括所有App混合方案,包括WebView方案、其他语言2Native方案、等。
随着技术的发展,有新的技术替代
桥协议
,如:JSI、等。
桥协议
:JS和C++互相无感知,只能通过桥协议
作为中间层,异步进行序列化/反序列化传输通讯。- JSI:将C++中的常用类型、定义的对象和函数 映射到JS中,支持JS随时调用C++中方法,有更高的性能、更低的通信开销(不需要再
异步进行序列化/反序列化传输通讯,可以同步直接调用方法、传输多种类型数据)。并且支持其他JS引擎。
postMessage
实现通讯的方案,不影响底层实现是:桥协议
的异步进行序列化/反序列化传输通讯 或 JSI实现。约定:本篇内容中描述
桥协议
的部分,都可以用这些新技术替代。
JS引擎:一个专门处理JS脚本的虚拟机
手机中的JS引擎:
- Android运用的JS引擎:Google的V8。
- iOS运行的JS引擎:Apple的JavaScriptCore。
还有其他JS引擎:SpiderMonkey、Rhino、Hermes等。
Hybrid底层依赖Native提供的容器(WebView),上层使用HTML、CSS、JS进行业务开发。
小程序是WebView+Native的双线程模型。
-
互相调用:
- Native调用WebView的JS方法(
window.前端定义方法
) - WebView调用
桥协议
(window.客户端定义方法
)、或触发自定义URL Scheme
(myscheme://客户端定义路径
)
- Native调用WebView的JS方法(
-
资源访问机制
-
以本地协议
file
方式访问Native内部资源。- 可以把前端要用的静态资源放到客户端本地(如:字体文件),本地页面通过类似
file:///android_asset/fonts/myFont.ttf
引用。
- 可以把前端要用的静态资源放到客户端本地(如:字体文件),本地页面通过类似
-
以远程
URL
方式访问线上资源(http/https)。 -
增量替换机制(不依赖发包更新)
- Native包内下载、解压线上的打包资源,再替换旧资源。
manifest
-
URL限定,限制访问、跨域问题的解决方案
- 可以限制WebView的能发起的请求内容。
- 可以代替WebView进行会触发跨域的AJAX请求。
-
-
页面在客户端内打开方式
-
针对产品功能性页面:
用本地协议
file
方式打开客户端包内.html(.js、.css、图片等都在客户端包内)。file
打开的页面直接发起请求可能会有跨域问题,可以用客户端接口代理的方式请求服务端数据。
-
针对运营活动页面:
用远程
URL
方式请求。
-
-
身份验证机制
Native创建WebView时,根据客户端登录情况注入跟登录有关的信息(session_id或token)至WebView(可注入到全局对象或cookie)。
cookie有同源策略,所以客户端可以对指定白名单域名添加cookie。
-
开发测试
- 提供切换成线上资源请求方式的功能,用代理工具代理成本地资源。
-
与Native通信方式:
-
桥协议
(JSI没有这个限制)都是以字符串(数据用JSON字符串)的形式交互,向客户端传递:- 全局的方法名 -> 客户端调用
window.方法名(JSON数据)
- 匿名函数 -> 客户端调用
(匿名函数(JSON数据))
- 全局的方法名 -> 客户端调用
-
WebView无法判断是否安装了其他App。
-
可以通过
查看注入的全局方法
、或客户端调用回调函数
(、或navigator.userAgent
)来判定页面是否在具体App内打开。 -
桥协议
仅在App内部起作用;自定义URL Scheme
是系统层面,所以可以额外针对跨App起作用(如:分享去其他App);iOS的通用链接可以认为是高级的自定义URL Scheme
。 -
桥协议
、自定义URL Scheme
方式的Native和WebView交互需要时间,对时效性很高的操作会有问题;JSI认为是同步的。 -
对前端操作(如:跳转
自定义URL Scheme
、全局方法调用)的监听拦截,都需要和客户端确认是否支持,以客户端描述为准。
-
桥协议
:Native注入全局方法至WebView的window
,若WebView调用则触发Native行为。- 客户端注入方式:javascript伪协议方式
javascript: 代码
。 - 注入JS代码可以在创建WebView之前(
[native code]
)或之后(全局变量JS注入)。
- 若注入的方法为
undefined
,则认为不在此App内部。
- 客户端注入方式:javascript伪协议方式
-
自定义URL Scheme
:监听拦截跳转(<iframe>
或<img>
设置src
、点击<a href>
、调用window.location.href =
),触发Native行为。URL Scheme
是iOS和Android提供给开发者的一种WAP唤起Native App方式(客户端用DeepLink、Deferred Deeplink实现)。Android应用在mainfest中注册自己的Scheme;iOS应用在App属性中配置。典型的URL Scheme:
myscheme://my.hostxxxxxxx
。- 客户端可以监听拦截任何行为(如:
console
、alert
)。相对于注入全局变量,监听拦截方式可以隐藏具体JS业务代码,且不会被重载,方便针对不可控的环境。 - 有些App会设置允许跳转的其他App的白名单或黑名单(记录其他APP的Scheme),如:微信白名单。
- 除了增加回调函数且被客户端调用,否则无法准确判定是否在此App内部。
- 跨App使用
自定义URL Scheme
,其后面的字符串要产生的行为仅目的App能理解。 - 快速触发多次
自定义URL Scheme
,有时仅有最后一个产生效果。e.g. 用window.location.href
快速触发多次,仅有最后一次跳转信息能够传递给客户端。 - 想要实现这样的功能:"关闭当前WebView,然后执行某功能",若用
window.location.href
等触发自定义URL Scheme
方式给客户端监听拦截,则可能WebView实例关闭后,跳转无法发出、因此无法被客户端获取(类似:异步的XMLHttpRequest
会随WebView实例关闭而被忽略)。可以尝试换用桥协议
。
-
JS触发
-
iOS8-支持
<iframe src="自定义URL Scheme">
和window.location.href="自定义URL Scheme"
跳转;iOS9+不支持,仅支持<iframe>
window.location.href="自定义URL Scheme"
跳转 -
iOS9+的Universal links(通用链接),可以从底层打开其他App客户端,跳过白名单(微信已禁用),若未安装则打开配置好的落地页(落地页再提示下载App,不需要
JS判断是否页面被隐藏从而跳转落地页)需要HTTPS域名配置、iOS设置等其他端配合。
参考:通用链接(Universal Links)的使用详解、Universal Link 前端部署采坑记、Support Universal Links。
-
Android支持
<iframe src="自定义URL Scheme">
和window.location.href="自定义URL Scheme"
跳转
-
使用自定义URL Scheme唤起APP,若页面没有被隐藏则认为未安装App,从而重定向至落地页/下载地址
-
方法一(推荐):
// 尝试跳转URL Scheme location.href = '自定义URL Scheme'; // 若支持则也可以用`<iframe>` const delay = 2000; // 按实际情况设置2~5秒 const timer = setTimeout(() => { location.href = '落地页/下载地址'; removeEventListener(); }, delay); // visibilitychange 事件方法(webkitvisibilitychange - webkitHidden,mozvisibilitychange - mozHidden) function clearFunc1() { if (document.hidden) { clearTimeout(timer); removeEventListener(); } } // pagehide 事件方法 function clearFunc2() { clearTimeout(timer); removeEventListener(); } function removeEventListener() { window.removeEventListener('visibilitychange', clearFunc1); window.removeEventListener('pagehide', clearFunc2); } // 若检测到隐藏页面,则阻止跳转落地页/下载地址(认为App已安装、已成功跳转URL Scheme) window.addEventListener('visibilitychange', clearFunc1); window.addEventListener('pagehide', clearFunc2);
-
方法二:
原理:浏览器非激活时,定时器执行时间会变慢/主线程被占用,所以会大于定时器时间之后才执行定时器内回调。
// 尝试跳转URL Scheme location.href = '自定义URL Scheme'; // 若支持则也可以用`<iframe>` const delay = 2000; // 按实际情况设置2~5秒 const start = Date.now(); setTimeout(function () { if (Date.now() - start < delay + 100) { // 还在这个页面,认为没有安装App location.href = '落地页/下载地址'; } }, delay);
-
落地页跳转App下载地址,除了iOS的企业包要注意用
itms-services://?action=download-manifest&url=网站链接.plist
之外,其他情况都可以直接打开下载地址。 -
-
在WebView中通过应用宝页面下载/打开其他APP
-
iOS:应用宝页面支持跳转至目标APP的App Store
-
Android:应用宝页面支持下载/打开目标APP
- 在腾讯系APP中能识别是否安装了目标APP;在其他APP中无法判断。
- 填写
android_schema
可以传递信息至目标APP:腾讯系APP打开目标APP后会带着信息;其他APP会先触发一次信息。
- 不用安装应用宝就支持打开目标APP功能;下载是去应用宝下载(会先要求安装应用宝APP)。
直接打开在聊天界面的URL,会有诸多限制,如:无法拉起APP、分享的URL不是卡片而是纯文本 等。打开 分享的卡片、二维码扫描 的网址,才有更多页面权限。
-
拼接应用宝下载/打开目标APP的链接:
-
应用宝主链接:
https://a.app.qq.com/o/simple.jsp?
-
跳转参数(search值,在
?
后面,用&
分割):-
APP包名:
pkgname=
+com.xx.xxx
-
渠道包链接(可选):
ckey=
+CK1234567890123
-
目标APP内打开路径(可选):
android_schema=
+自定义URL Scheme://具体跳转路径
若有一些特殊字符,则可以用
encodeURIComponent
转义属性名和属性值。
-
e.g.
https://a.app.qq.com/o/simple.jsp?pkgname=com.xx.xxx&ckey=xxxx&android_schema=xxxx://xx
-
-
-
落地页按钮方案
-
方案一:
仅一个「打开App」按钮,包含先尝试打开App、然后尝试下载App(使用自定义URL Scheme唤起APP,若页面没有被隐藏则认为未安装App,从而重定向至下载地址)
e.g. https://kg.qq.com/
-
方案二:
一个按钮仅「打开App」,一个按钮仅「下载App」
e.g. 支付宝App
- pc端下载页:https://render.alipay.com/p/yuyan/180020040001212700/
- 移动端下载页: https://d.alipay.com(去除打开App按钮的下载页:https://d.alipay.com/?nojump=true)
- 「下载App」考虑是否跳转到应用中心App(支持更新、下载、打开,但一个时刻仅支持其中一种功能)
-
- 客户端可以监听拦截任何行为(如:
-
WebView提供给Native调用的全局回调函数(或匿名函数)。
对于动态创建的全局回调函数,要注意同名覆盖问题
e.g.
let _localCounter = 1 // 同一个方法名快速请求时,可能 Date.now() 还没有变化 function invokeJSBridge (method, arg, { hasCallback = true }) { return new Promise((resolve, reject) => { if (typeof window.客户端定义方法 === 'function') { let callbackName = '' // 创建客户端回调 if (hasCallback) { callbackName = `${method}CallbackName_${_localCounter}` _localCounter += 1 window[callbackName] = (res) => { // fixme: 增加定时器处理长时间未被客户端回调的方法 try { resolve({ result: 'ok', data: JSON.parse(res) }) } catch (e) { resolve({ result: 'error', data: res }) } window[callbackName] = null } } // 调用客户端方法 window.客户端定义方法( // 桥协议 method, // 方法名 JSON.stringify(arg || {}), // 传参 callbackName // 回调 ) // 无客户端回调时直接完成 if (!hasCallback) { resolve({ result: 'ok', data: '' }) } } else { reject(arg) } }) } /* 使用测试 */ invokeJSBridge('方法名', '参数') .then((res) => { // 是客户端、且调用成功&&客户端执行回调 // 根据res处理客户端执行之后业务 }) .catch((res) => { // 不是客户端 // 非客户端业务 })
接口设计可以带有「透传数据」:前端调用客户端方法时多传一个透传参数,之后客户端异步调用前端方法时带着这个参数的值。
-
以 全双工通讯协议 作为桥梁通信,如:前端的websocket,客户端启动在后台的通信服务。
-
RN前端和
<WebView>
组件通信:postMessage
。postMessage
实现通讯的方案,不影响底层实现是:桥协议
的异步进行序列化/反序列化传输通讯 或 JSI实现。
-
-
根据WebView的错误处理机制统计用户在WebView遇到的bug。
-
分享到其他App
- 通过JS触发Native App之间的切换分享(自己Native内可用
桥协议
,任意App均要起作用只能用自定义URL Scheme
)。 - 带分享信息参数去访问目标App提供的分享URL。
- 通过JS触发Native App之间的切换分享(自己Native内可用
-
客户端内的前端页面,添加半屏弹窗策略:
针对通用的逻辑,要给多个前端页面添加半屏弹窗逻辑。如:某个后台接口返回错误码,统一弹出半屏弹窗。
- 提前注入弹窗DOM和逻辑,需要触发时调用。
- 需要触发时动态注入弹窗DOM和逻辑,再调用。
- 新制作一个前端的弹窗页面,需要触发时在原页面上层覆盖这个半屏弹窗页面。
- 客户端提供bridge或Scheme用于弹出客户端的弹窗,需要触发时唤起客户端的弹窗。
-
js调用App的中间层代码,兼容实现
不同平台、不同App、环境准备成功与否
的逻辑- 支持:主动调用、事件通知、URL scheme
- 支持:新App低成本接入
-
限制js调用的入参和出参(回调函数的类型 或 返回Promise的类型),参数结构统一
- 协议规范(强类型)
- 客户端返回的错误码规则(错误码决定错误类型,如:调用未注册接口、导致crash、参数问题)
- js错误兜底(
try-catch
、超时时间),错误码处理(如:提示版本问题、提示升级、静默失败无反馈)
-
接口文档,如何根据代码自动生成(jsdoc、typedoc)
- 接口demo调试页面 支持 页面实时运行的代码编辑体验(参考ant-design:利用sandpack)
-
*3端之间 降低沟通成功、提高变更同步效率
控制唯一的修改入口(前端、iOS、Android,仅选其一),由某一端人员定义协议,自动生成3端接口代码
-
*接口成功率监控、告警,异常上报
-
*流程接入工程化
-
移动端技术演进:Native -> Hybrid -> Native容器 -> 自绘
-
Native App
- 高成本开发
- 原生性能体验
- 依赖发包更新
-
WebView的页面
- 低成本开发、高效率、一套代码跨平台复用(因为就是输出页面)
- 性能不及原生应用(因为要初始化WebView,前端「增量」原则决定)
- 不依赖Native发包更新(注意避免违反苹果的热更新条款)
-
Native容器方案
以Weex(已掉队)和RN最具代表,原理类似,上层JS利用Vue/React框架维护虚拟节点,通过bridge(或JSI等优化)与Native通信,最终在Native渲染组件。
Native容器方案 与 自绘方案,就目前而言没有谁优谁劣:对于Native容器方案来说,更多的是代表了研发效率与动态性;对于Flutter这类框架而言,虽然实现了高性能、多端一致的自绘,但却缺少一套较为完备的动态化方案。
-
自绘方案
客户端只需提供一个画布,框架自带渲染引擎在画布上进行UI渲染和逻辑(不需要客户端帮忙渲染,因此不需要RN那样的一层Bridge进行JS和Native的通信数据转发,渲染效率更高)。
e.g. Flutter:既不使用WebView,也不使用操作系统的原生控件。原生系统提供2D画布(
Canvas
),Flutter使用自己的高性能渲染引擎Skia在画布上绘制Widget。这样不仅可以保证在Android和iOS上UI的一致性,而且可以避免因对原生控件依赖而带来的限制及高昂的维护成本。
-
-
WebView启动流程
-
相对于Native App的流畅体验,WebView的页面瓶颈一般都卡在WebView实例初始化,可能导致App卡顿、页面加载缓慢。
WebView的初始化、保持,占用较多内存。
-
与浏览器不同,App中打开WebView的第一步并不是建立连接,而是启动浏览器内核。
-
除了网站性能优化之外,能提升体验的办法基本要让Native配合调整(页面能做的不多,主要靠客户端开发投入,具体方案可查看:WebView性能、体验分析与优化)。
-
前端编写的代码,通过中间的其他语言2Native
之类的方式(bridge)转换为原生App代码运行,原生App代码也会通过Native2其他语言
(bridge)回调前端编写的代码。
如:Dart2Native、React Native2Native、Weex2Native。