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

2019-06-0319:41:59APP与小程序开发Comments2,261 views字数 1721阅读模式

开发微信小程序的时候,获取用户信息、openid还有地理位置这些信息的时候,都是采用Promise的方式异步获取,但是这样的话在页面和App.js中都获取就可能造成请求重复的问题。文章源自菜鸟学院-https://www.cainiaoxueyuan.com/xcx/13292.html

比如为了在每个页面都能获取到这些共享信息,都会选择在App.js中进行获取,然后在页面级进行获取,这两次获取的时间间隔较小时就可能导致前一个请求还未获取到数据,后一个请求就会再次进行获取,这样就产生了两次请求。文章源自菜鸟学院-https://www.cainiaoxueyuan.com/xcx/13292.html

还有一个问题就是书写麻烦(虽然也能通过async await简化),比如文章源自菜鸟学院-https://www.cainiaoxueyuan.com/xcx/13292.html

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

正好周末的时候突然想到了vue的watch语法,利用一些相关的知识,就可以解决这个麻烦的问题了。文章源自菜鸟学院-https://www.cainiaoxueyuan.com/xcx/13292.html

解决思路

双向绑定

vue的双向绑定原理,3.0将会采用Proxy监听数据变化,不过考虑到小程序这边的Proxy兼容性我不知道,所以采用了2.0的Object.defineProperty来监听数据的变化。文章源自菜鸟学院-https://www.cainiaoxueyuan.com/xcx/13292.html

主要还是拦截设置的操作,在进行赋值时,将新旧值通知至监听者。文章源自菜鸟学院-https://www.cainiaoxueyuan.com/xcx/13292.html

观察者模式

在页面级的onLoad监听app.globalData各个键名的事件,而在app.js的onLoad中则使用Object.defineProperty重新定义app.globalData,这样一旦app.globalData相应的键值发生了变化,就会通知监听的页面该值发生了变化。文章源自菜鸟学院-https://www.cainiaoxueyuan.com/xcx/13292.html

模块化的引用

观察者模式导出的是一个对象(类实例),而不是一个类,所以在导入的时候这个对象是共享的,就可以通过这个对象将app.js和其他页面联系起来。文章源自菜鸟学院-https://www.cainiaoxueyuan.com/xcx/13292.html

至于模块加载的实质,ES6模块加载的机制,与CommonJS模块完全不同。感兴趣的可以去看看这个文章源自菜鸟学院-https://www.cainiaoxueyuan.com/xcx/13292.html

封装Page

小程序的Page函数本身是不支持watch,但是我们可以自定义一个函数,进行参数合并就可以了。文章源自菜鸟学院-https://www.cainiaoxueyuan.com/xcx/13292.html

在页面onLoad时先遍历watch属性,对app.globalData进行监听,可以参考vue的watch用法。文章源自菜鸟学院-https://www.cainiaoxueyuan.com/xcx/13292.html

页面onUnload时就会进行销毁,此时也应该取消监听,这些我都封装过了,不用手动处理了。文章源自菜鸟学院-https://www.cainiaoxueyuan.com/xcx/13292.html

有了这些思路,用不了多久,一个雏形就出来了,经过手动测试,感觉没什么问题,我就发布到npm了,大家感兴趣的可以安装体验一下。文章源自菜鸟学院-https://www.cainiaoxueyuan.com/xcx/13292.html

安装

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…文章源自菜鸟学院-https://www.cainiaoxueyuan.com/xcx/13292.html

作者:ma125120
链接:https://juejin.im/post/5cf479e86fb9a07eda03013e
来源:掘金
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。文章源自菜鸟学院-https://www.cainiaoxueyuan.com/xcx/13292.html

文章源自菜鸟学院-https://www.cainiaoxueyuan.com/xcx/13292.html
  • 本站内容整理自互联网,仅提供信息存储空间服务,以方便学习之用。如对文章、图片、字体等版权有疑问,请在下方留言,管理员看到后,将第一时间进行处理。
  • 转载请务必保留本文链接:https://www.cainiaoxueyuan.com/xcx/13292.html

Comment

匿名网友 填写信息

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定