如何通过Proxy实现JSBridge模块化封装


Posted in Javascript onOctober 22, 2020

最近公司在做一个项目,通过把我们自己的Webview植入第三方APP,然后我们的业务全部通过H5实现。至于为什么不直接用第三方APP WebView,主要是身处金融行业,需要做一些风控相关功能。

由于是Hybrid APP的性质,所以web与Native的通信是无法避免的;而为什么我要封装jsBridge,主要在于下面两点:

公司APP的JSBridge提供了数据的序列化和全局函数的注入,而我们这次由于包大小考虑,这一块需要H5自己来实现;

原生提供的接口协议太多,记住麻烦;

回调的写法不太人性化,期望Promise;

由于本次项目只涉及到Andriod,所以没有关于ios的处理,但我自认为他们只是协议的不同,Web的处理可以相同。

原理浅谈

如何通过Proxy实现JSBridge模块化封装

看上图的通信实现(图片来源于文章开头的文章),简单说一下通信过程;

Webview加载时会将原生提供的JSBridge方法注入到window对象上,比如:window.JSBridge.getDeviceInfo就是原生提供的可以读取一些设备标识信息的接口;

H5通过window调用原生接口,基本都需要传参,比如这次处理成功或则处理失败的结果回调的,还有一些参数设置,拿上面给的方法来举例:

window.JSBridge.getDeviceInfo({
 token: '*&^%$$#*',
 onOk(data) {
  save(data);
 },
 onError(error) {
  console.log(error.message);
 }
});

原生响应H5的调用成功或失败后,就执行H5传递过来的回调函数;

过程结束;

看上面的通信过程,貌似很简单。但这里面存在一些协议的问题:

首先H5与原生端的通信消息,是只支持字符串的,如果要传JS对象,那就先序列化;

序列化带来的后果又是,对象中的函数就无法传递;

而就算函数传过去了,也是存在问题的,由于安全的限制,webview和js的执行没有在一个容器中,回调这种局部函数是找不到的,所以是需要将回调函数注册到全局;

所以下面就来解决这些问题

一步一步的具体实现

接口协议封装

什么意思喃?看下面的图:

如何通过Proxy实现JSBridge模块化封装

由于APP端协议及分包问题, 存在多个Bridge, 比如MBDevice、MBControl、MBFinance,上面列出来的只是一小部分,对于web来说记忆这些接口是一件很费事的事;还有就是以前我调APP的JSBridge, 总有下面这样的代码:

window.JSBridge && window.JSBridge.getDeviceInfo && window.JSBridge.getDeviceInfo({ ... })

至于上面,所以加了一层封装,实现的核心就是Proxy和Map,具体实现看下面的伪代码:

const MBSDK = {
};

// sdk 提供的方法白名单
const whiteList = new Map([
 ['setMaxTime', 'MBVideo'],
 ['getDeviceInfo', 'MBDevice.getInfo'],
 ['close', 'MBControl'],
 ['getFinaceInfo', 'MBFinance.getInfo'],
]);

const handler = {
 get(target, key) {
  if (!whiteList.has(key)) {
   throw new Error('方法不存在');
  }
  const parentKey = whiteList.get(key);
  function callback() {
   return [...parentKey.split('.'), key];
  }
  return new Proxy(callback, applyHandler); // funcHandler后面再展开
 },
};
export default new Proxy(MBSDK, handler);

基于上面的封装,调用时,代码就是下面这样

sdk.setMaxTime({
   maxTime: 10,
  }).then(() => {
   console.log('设置成功');
  }, () => {
   window.alert('调用失败');
  });

序列化与回调注册

上面已经列了为什么需要回调函数全局注册和序列化,这里主要说一下实现原理,总得来说分两步;

回调函数剥离,全局注册;

参数序列化;

回调函数剥离和参数序列化

其实很好实现,直接展开运算符搞定:

const { onOk, onError, ...others } = params; // 回调函数剥离
const str = JSON.stringify(others); // 参数序列化

函数全局注册

看了很多文章的一些实现,思路基本一致,比如下面这样

window.bridgeCallbacks = {};
const callBacks = window.bridgeCallbacks;
const { onOk, onError, ...others } = params; // 回调函数剥离

const callbackId = generateId(); // 产生一个唯一的随机数Id

callBacks[`success_${callbackId}`] = onOk;
callBacks[`onError${callbackId}`] = onError;

others.success = `window.bridgeCallbacks.success_${callbackId}`
// ....
// 调用jdk代码

这是一种很容易想到的问题,但却存在一些问题,比如:

bridgeCallbacks全局会注册很多属性,因为Native调用并没有清理,而onOk这种很多时候是一个闭包,由于有引用,最后导致的问题就是内存泄露;

就算处理了第一步的问题,webview无响应怎么办,那回调就会被一直挂起,确少超时响应逻辑

callbackId的唯一性不好保证;

基于以上考虑,我换了一个方案,采用回调队列,因为APP端说过,回调是按顺序的,不会插队;

class CallHeap {
 constructor() {
  this.okQueue = [];
  this.errorQueue = [];
 }
 success = (args) => {
  // 成对弹出回调:成功时,不止要处理成功的回调,失败的也要同时弹出,
  const target = this.okQueue.shift();
  this.errorQueue.shift();
  target && target(args);
 }
 error = (args) => {
  const target = this.errorQueue.shift();
  this.okQueue.shift();
  target && target(args);
 }
 addQueue(onOk = Null, onError = Null) {
  this.okQueue.push(onOk);
  this.errorQueue.push(onError);
 }
}

window.bridgeCallbacks = {};
const callBacks = window.bridgeCallbacks;

function applyhandler() {
 const { onOk, onError, ...others } = params; // 回调函数剥离
 if (onOk || onError) {
   const callKey = transferKey || key; // transferKey || key后面会提到
   // 如果全局未注册,则先注册对应的调用域
   if (!callbacks[callKey]) {
    callbacks[callKey] = new CallHeap();
   }
   // 添加回调
   callbacks[callKey].addQueue(onOk, onError);

   others.success = `callBacks.${callKey}.success`;
   others.error = `callBacks.${callKey}.error`;
  }
  // 调用jdk代码
}

基于以上的实现,就可以保证发起多个Native请求,并保证有序回调;如果成功,成功回调被响应时,响应的失败回调也会被弹出,因为回调函数式存在数组中的,所以执行完后,引用就不会再存在。

完整实现

看了上面的代码实现,但核心好像还没有提及,那就是调用参数的拦截。前面我们用Proxy的get优雅的实现了SDK方法的拦截,这里会接着采用Proxy的apply方法来拦截方法调用的传参,直接看代码吧:

// 结合最上面接口协议封装的代码一起看
const applyHandler = {
 apply(target, object, args) {
  // transferKey 用于getFinaceInfo与getDeviceInfo这种数据命名重复的
  const [parentKey, key, transferKey] = target();
  console.log('res', parentKey, key);
  const func = (SDK[parentKey] || {})[key];

  const { onOk, onError, ...params } = args[0] || {};

  if (onOk || onError) {
   const callKey = transferKey || key;
   if (!callbacks[callKey]) {
    callbacks[callKey] = new CallHeap();
   }
   callbacks[callKey].addQueue(onOk, onError);

   others.success = `callBacks.${callKey}.success`;
   others.error = `callBacks.${callKey}.error`;
  }

  return func && (window[parentKey][key])(JSON.stringify(params));;
 }
};

Promise 封装

前面吹过的牛逼还有两个没实现,比如:

promise支持

超时调用

首先来复习一下,怎么封装一个支持Promise的setTimeout函数:

function promiseTimeOut(time) {
 return new Promise((resolve, reject) => {
  setTimeout(resolve, time);
 });
}

promiseTimeOut(1000).then(() => {
 console.log('time is ready');
})

如果对上面这个封装不陌生,那基于回调函数的Promise化就变得简单了

talk is cheap, show me your code

完整实现:

const MBSDK = {
};

// sdk 提供的方法白名单
const whiteList = new Map([
 ['setMaxTime', 'MBVideo'],
 ['getDeviceInfo', 'MBDevice.getInfo'],
 ['close', 'MBControl'],
 ['getFinaceInfo', 'MBFinance.getInfo'],
]);

const applyHandler = {
 apply(target, object, args) {
  // transferKey 用于getFinaceInfo与getDeviceInfo这种数据命名重复的
  const [parentKey, key, transferKey] = target();
  // FYX 编程
  const func = (window[parentKey] || {})[key];
  // 设置一个默认的超时参数,支持配置
  const { timeout = 5000, ...params } = args[0] || {};

  return new Promise((resolve, reject) => {
   const callKey = transferKey || key;
   if (!callbacks[callKey]) {
    callbacks[callKey] = new CallHeap();
   }
   const timeoutId = setTimeout(() => {
    // 超时,主动发起错误回调
    window.callBacks[callKey].error({ message: '请求超时' });
   }, timeout);
   callbacks[callKey].addQueue((data) => {
    clearTimeout(timeoutId);
    resolve(data);
   }, (data) => {
    clearTimeout(timeoutId);
    reject(data);
   });
   params.success = `callBacks.${callKey}.success`;
   params.error = `callBacks.${callKey}.error`;
   func && (window[parentKey][key])(JSON.stringify(params));
  }).catch((error) => {
   console.log('error:', error.message);
  });
 }
};

const handler = {
 get(target, key) {
  if (!whiteList.has(key)) {
   throw new Error('方法不存在');
  }
  const parentKey = whiteList.get(key);
  function callback() {
   return [...parentKey.split('.'), key];
  }
  return new Proxy(callback, applyHandler); // funcHandler后面再展开
 },
};

export default new Proxy(MBSDK, handler);

而调用时,基本上,就可以这样玩了:

sdk.setMaxTime({
   maxTime: 10,
  }).then(() => {
   console.log('设置成功');
  }, () => {
   window.alert('调用失败');
  });

解惑

- func.call(null, JSON.stringify(params)) // 以前的

+ func && (window[parentKey][key])(JSON.stringify(params)); // 现在的

开始函数的调用是采用func.call来实现的,当时我本地mock过,没有问题。但在webview中就弹出了下面这样一个错误:

java bridge method can't be invoked on a non-injected object

经过各种goggle,百度,查到的都是一条关于Andriod的注入漏洞。而至于我这里通过JS的方式把bridge指向的函数地址,赋值给一个变量名,然后再通过变量名来调用就会报上面这个错误,我个人的猜测有两个:一是协议这样规定的;二是this指向问题。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
escape、encodeURI 和 encodeURIComponent 的区别
Mar 02 Javascript
javascript简易缓动插件(源码打包)
Feb 16 Javascript
jquery的选择器的使用技巧之如何选择input框
Sep 22 Javascript
java和javascript获取word文档的书签位置对比
Jun 19 Javascript
JavaScript_ECMA5数组新特性详解
Jun 12 Javascript
jQuery Ajax File Upload实例源码
Dec 12 Javascript
bootstrap警告框使用方法解析
Jan 13 Javascript
Vue.2.0.5过渡效果使用技巧
Mar 16 Javascript
js实现随机点名系统(实例讲解)
Oct 18 Javascript
浅谈在node.js进入文件目录的问题
May 13 Javascript
前端性能优化建议
Sep 17 Javascript
JavaScript+HTML实现学生信息管理系统
Apr 20 Javascript
微信小程序canvas动态时钟
Oct 22 #Javascript
vue-cli —— 如何局部修改Element样式
Oct 22 #Javascript
微信小程序入门之绘制时钟
Oct 22 #Javascript
微信小程序入门之指南针
Oct 22 #Javascript
微信小程序实现拼图小游戏
Oct 22 #Javascript
Vue select 绑定动态变量的实例讲解
Oct 22 #Javascript
在Vue中使用Select选择器拼接label的操作
Oct 22 #Javascript
You might like
PHP CURL模拟登录新浪微博抓取页面内容 基于EaglePHP框架开发
2012/01/16 PHP
如何取得中文字符串中出现次数最多的子串
2013/08/08 PHP
php5.3以后的版本连接sqlserver2000的方法
2014/07/28 PHP
JavaScript与Image加载事件(onload)、加载状态(complete)
2011/02/14 Javascript
google jQuery 引用文件,jQuery 引用地址集合(jquery 1.2.6至jquery1.5.2)
2011/04/24 Javascript
js获取RadioButtonList的Value/Text及选中值等信息实现代码
2013/03/05 Javascript
用原生JavaScript实现jQuery的$.getJSON的解决方法
2013/05/03 Javascript
选择器中含有空格在使用示例及注意事项
2013/07/31 Javascript
js控制淡入淡出示例代码
2013/11/12 Javascript
javascript中Math.random()使用详解
2015/04/15 Javascript
JS实现上下左右对称的九九乘法表
2016/02/22 Javascript
Bootstrap组件(一)之菜单
2016/05/11 Javascript
Javascript for in的缺陷总结
2017/02/03 Javascript
jQuery实现的页面弹幕效果【测试可用】
2018/08/17 jQuery
移动端H5页面返回并刷新页面(BFcache)的方法
2018/11/06 Javascript
Vue 实现前端权限控制的示例代码
2019/07/09 Javascript
JS删除对象中某一属性案例详解
2020/09/08 Javascript
Python 检查数组元素是否存在类似PHP isset()方法
2014/10/14 Python
python操作 hbase 数据的方法
2016/12/18 Python
python寻找list中最大值、最小值并返回其所在位置的方法
2018/06/27 Python
在python中实现将一张图片剪切成四份的方法
2018/12/05 Python
Python批量修改图片分辨率的实例代码
2019/07/04 Python
Python使用scipy模块实现一维卷积运算示例
2019/09/05 Python
Html5调用手机摄像头并实现人脸识别的实现
2018/12/21 HTML / CSS
一百多行代码实现react拖拽hooks
2021/03/23 Javascript
机电专业个人求职信范文
2013/12/30 职场文书
咖啡厅创业计划书范本
2014/01/22 职场文书
《蜗牛》教学反思
2014/02/18 职场文书
药学职务聘任书
2014/03/29 职场文书
机电专业毕业生自我鉴定2014
2014/10/04 职场文书
工伤事故赔偿协议书
2014/10/27 职场文书
初中生考试作弊检讨书
2014/12/14 职场文书
企业党建工作总结2015
2015/05/26 职场文书
2016年教师学习廉政准则心得体会
2016/01/20 职场文书
MySQL笔记 —SQL运算符
2022/01/18 MySQL
Eclipse+Java+Swing+Mysql实现电影购票系统(详细代码)
2022/01/18 Java/Android