使用watch在微信小程序中实现全局状态共享


Posted in Javascript onJune 03, 2019

问题

在之前开发微信小程序的时候,获取用户信息、openid还有地理位置这些信息的时候,都是采用Promise的方式异步获取,但是这样的话在页面和App.js中都获取就可能造成请求重复的问题。

比如为了在每个页面都能获取到这些共享信息,都会选择在App.js中进行获取,然后在页面级进行获取,这两次获取的时间间隔较小时就可能导致前一个请求还未获取到数据,后一个请求就会再次进行获取,这样就产生了两次请求。

还有一个问题就是书写麻烦(虽然也能通过async await简化),比如

onLoad() {
 app.getUserInfo()
 .then(userInfo => {
 
 }).catch(err => { /* 错误处理 */ });
 
 // 如果同时需要userInfo和openid,可能就是如下形式:
 Promise.all([app.getUserInfo(), app.getOpenid()])
 .then(res => {
 
 }).catch(err => { /* 错误处理 */ });
}

正好周末的时候突然想到了vue的watch语法,利用一些相关的知识,就可以解决这个麻烦的问题了。

解决思路

双向绑定

vue的双向绑定原理,3.0将会采用Proxy监听数据变化,不过考虑到小程序这边的Proxy兼容性我不知道,所以采用了2.0的Object.defineProperty来监听数据的变化。

主要还是拦截设置的操作,在进行赋值时,将新旧值通知至监听者。

观察者模式

在页面级的onLoad监听app.globalData各个键名的事件,而在app.js的onLoad中则使用Object.defineProperty重新定义app.globalData,这样一旦app.globalData相应的键值发生了变化,就会通知监听的页面该值发生了变化。

模块化的引用

观察者模式导出的是一个对象(类实例),而不是一个类,所以在导入的时候这个对象是共享的,就可以通过这个对象将app.js和其他页面联系起来。

至于模块加载的实质,ES6模块加载的机制,与CommonJS模块完全不同。感兴趣的可以去看看这个。

封装Page

小程序的Page函数本身是不支持watch,但是我们可以自定义一个函数,进行参数合并就可以了。

在页面onLoad时先遍历watch属性,对app.globalData进行监听,可以参考vue的watch用法。

页面onUnload时就会进行销毁,此时也应该取消监听,这些我都封装过了,不用手动处理了。

有了这些思路,用不了多久,一个雏形就出来了,经过手动测试,感觉没什么问题,我就发布到npm了,大家感兴趣的可以安装体验一下。

安装

npm i wx-watch -S --production

使用

// app.js
var { watchData, } = require('/miniprogram_npm/wx-watch/index.js');

App({
 onLaunch() {
 this.watchData(); /* 监听this.globalData的变化,并触发事件,其他页面监听的值必须在globalData中预先定义,否则无法监听 */
 },
 watchData,
 globalData: {
 userInfo: null,
 }
});

// 其他需要监听globalData的页面.js
var { getPage } = require('../../miniprogram_npm/wx-watch/index.js');
const app = getApp();

/**
 * getPage(页面参数,app) app必传,因为封装的时候访问不到,就只能传参了
*/
getPage({
 watch: {
 userInfo(userInfo, oldUserInfo) {
 console.log(`来自app.glodalData的userInfo`);
 }
 },
 // 其他参数
}, app)

github:  github.com/ma125120/wx…

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,谢谢大家对三水点靠木的支持。

Javascript 相关文章推荐
jQuery实现的个性化返回底部与返回顶部特效代码
Oct 30 Javascript
jQuery实现ajax调用WCF服务的方法(附带demo下载)
Dec 04 Javascript
js判断图片加载完成后获取图片实际宽高的方法
Feb 25 Javascript
AngularJS 中文API参考手册
Jul 28 Javascript
JS弹出窗口的运用与技巧大全
Nov 01 Javascript
js实现的xml对象转json功能示例
Dec 24 Javascript
bootstrap中的 form表单属性role="form"的作用详解
Jan 20 Javascript
jQuery读取XML文件的方法示例
Feb 03 Javascript
angular directive的简单使用总结
May 24 Javascript
vue 1.0 结合animate.css定义动画效果
Jul 11 Javascript
react中实现搜索结果中关键词高亮显示
Jul 31 Javascript
Jquery的autocomplete插件用法及参数讲解
Mar 12 jQuery
深入理解JS异步编程-Promise
Jun 03 #Javascript
模块化react-router配置方法详解
Jun 03 #Javascript
react 组件传值的三种方法
Jun 03 #Javascript
angular使用md5,CryptoJS des加密的方法
Jun 03 #Javascript
Node.js 的 GC 机制详解
Jun 03 #Javascript
微信小程序蓝牙连接小票打印机实例代码详解
Jun 03 #Javascript
react-native滑动吸顶效果的实现过程
Jun 03 #Javascript
You might like
用mysql_fetch_array()获取当前行数据的方法详解
2013/06/05 PHP
PHP 实现类似js中alert() 提示框
2015/03/18 PHP
各种快递查询--Api接口
2016/04/26 PHP
firefox中JS读取XML文件
2006/12/21 Javascript
javascript模拟枚举的简单实例
2014/03/06 Javascript
JS辨别访问浏览器判断是android还是ios系统
2014/08/19 Javascript
JS中FRAME的操作问题实例分析
2014/10/21 Javascript
node.js中的fs.mkdir方法使用说明
2014/12/17 Javascript
BootStrap3学习笔记(一)之网格系统
2016/05/20 Javascript
jQuery UI Bootstrap是什么?
2016/06/17 Javascript
js实现四舍五入完全保留两位小数的方法
2016/08/02 Javascript
jquery根据一个值来选中select下的option实例代码
2016/08/29 Javascript
简易的JS计算器实现代码
2016/10/18 Javascript
使用纯JS代码判断字符串中有多少汉字的实现方法(超简单实用)
2016/11/12 Javascript
微信小程序分页加载的实例代码
2017/07/11 Javascript
浅谈vue中使用图片懒加载vue-lazyload插件详细指南
2017/10/23 Javascript
Vue中的无限加载vue-infinite-loading的方法
2018/04/08 Javascript
JS+HTML5本地存储Localstorage实现注册登录及验证功能示例
2020/02/10 Javascript
js实现页面图片消除效果
2020/03/24 Javascript
详解C++编程中一元运算符的重载
2016/01/19 Python
浅谈python中的正则表达式(re模块)
2017/10/17 Python
Python基于hashlib模块的文件MD5一致性加密验证示例
2018/02/10 Python
Python Django Vue 项目创建过程详解
2019/07/29 Python
基于TensorBoard中graph模块图结构分析
2020/02/15 Python
Python开发之pip安装及使用方法详解
2020/02/21 Python
浅谈对python中if、elif、else的误解
2020/08/20 Python
毕业证丢失证明
2014/01/15 职场文书
成龙洗发水广告词
2014/03/14 职场文书
协议书与合同的区别
2014/04/18 职场文书
群众路线剖析材料(四风问题)
2014/10/08 职场文书
2015年教研室工作总结范文
2015/05/23 职场文书
可怜妈妈观后感
2015/06/09 职场文书
2016元旦文艺汇演主持词(开场白+结束语)
2015/12/03 职场文书
2019辞职报告范本3篇!
2019/07/23 职场文书
利用python实时刷新基金估值(摸鱼小工具)
2021/09/15 Python
CSS使用SVG实现动态分布的圆环发散路径动画
2022/12/24 HTML / CSS