详解小程序开发经验:多页面数据同步


Posted in Javascript onMay 18, 2019

导语:本文主要介绍在小程序中,多页面之间如何保持数据同步

在很多的产品中,都会存在跨页面间需要数据同步,如下示例:

详解小程序开发经验:多页面数据同步

为了更好的理解该场景,我们再详细描绘一下:

  1. 本场景包括4个页面:动态广场、个人中心、我的动态、动态详情
  2. 首先,进入动态广场页,请求加载数据,展示动态列表,其中,我们用绿色内阴影区分该条动态是“我的”,其他未加内阴影的表示是“别人的”;
  3. 然后,进入个人中心页,请求加载数据,展示获赞数量;
  4. 点击我的动态,进入我的动态页,请求加载数据,展示我的动态列表;
  5. 点击其中一条动态,进入动态详情页,请求加载数据,进行点赞操作;
  6. 在第5步中,点赞成功后,回退到我的动态页,可以看到该条动态点赞状态和数量发生变化,已经同步;
  7. 再回到到个人中心页,也可以看到获赞数量发生变化,已经同步;
  8. 再回到动态广场页,也可以看到对应的一条动态点赞状态和数量发生变化,已经同步;

下面我们来探讨一下这个场景的实现,在此之前,我们先要了解在点赞时,该场景中各页面的状态及关系。

详解小程序开发经验:多页面数据同步

详解小程序开发经验:多页面数据同步

如上图所示,当我们在点赞时,4个页面都已经在是打开的(4个webview)。当我们点赞成功时,点击左上解返回时,动态详情页的webview关掉,直接看到下一层webview,也就是我的动态页,这个页面是已经存在的。其他页面也是如此。
那对于这些已经存在的页面,我们应该如何同步更新数据呢?

当然,如果比较懒,可以直接在onShow的时候重新拉数据渲染页面。但显然这是非常低级、不可取也没必要的做法。重新拉数据需要耗时,页面重新渲染也会看到闪屏,关键是根本没必要重新拉数据,因为数据发生了变化,前端是知道的。

所以我们可以这样做,在动态详情页点赞成功时,保存一个数据到全局globalData中去,回到我的动态页,在onShow中去检测全局globalData中是否有点赞变化的数据,有的话,就读取出来去更新相应的动态。

// 动态详情页js
onLike() {
 ...
 success: () => {
 App.globalData.like = {
  fid: 10001,
  likes: 1,
  hasLike: true   
 }
 }
}

// 我的动态页js
onShow() {
 if(App.globalData.like !== null) {
 // 读取globaldata.like数据去更新
 this.doUpdata()
 // 特别需要注意,更新完后,需要把globaldata.like清掉,不然下次onShow还会继续走到该逻辑
 App.globalData.like = null
 }
}

这样似乎可以达到我们的目的,无请求、纯前端局部更新。

但这样还存在一个问题,当我们再退回到个人中心页时,要检查下获赞数量是否需要更新,以及回到动态广场页时,也要检查点赞有没有发生变化。但在这两个页面onShow去判断App.globalData.like时,都已经检测不到了,因为该数据已经在我的动态页onShow中置为null了。

概括来说,在点赞时,只生产了一条数据,但有多个消费者,哪个页面先把数据消费了,其他页面也就无法检测到数据了。

由此,我们想到那就使用EventBus来处理。
首先,我们自己实现一套简单的EventBus。

源码见:git.weixin.qq.com/xinyuanliu/…

在小程序启动时,初始化EventBus:

const Event = require('/util/events.js').default

App({
 events: null,
 onLaunch(options) {
 this.initEvents()
 // doOtherThings
 },
 initEvents() {
 this.events = new Event()
 },
 emitFeedsLike(data) {
 this.events.emit('feedsLike', data)
 },
 emitPublishFeeds(data) {
 this.events.emit('publishFeeds', data)
 },
 ...
}

各个页面在onLoad时,注册监听事件(在此以我的动态页为例):

// 我的动态.js
const App = getApp()

Page({
 data: {
 list: []
 },
 onLoad: function (options) {
 ...
 // 监听点赞事件广播
 ↓ 重点在这里 ↓
 App.events.on('feedsLike', data => {
  console.log('我的动态页面收到点赞变化通知:', data)
  // 进行更新操作
 })
 // 监听发布事件广播
 ↓ 重点在这里 ↓
 App.events.on('publishFeeds', data => {
  console.log('我的动态页面收到发布动态通知:', data)
  // 进行更新操作
 })
 },
 ...
})

然后在动态点赞时,发出事件通知。(这里一条动态是封装成组件,不属于某一个页面,点赞事件也是封装在组件内)

const App = getApp()

Component({
 properties: {...},
 methods: {
 // 点赞
 tapLike(e) {
  let { likes, hasLike } = this.data

  likes += (hasLike && -1 || 1)
  hasLike = !hasLike

  this.updateFeeds(likes, hasLike).then(() => {
  this.setData({
   likes,
   hasLike
  })

  // 广播事件
  ↓ 重点在这里 ↓
  App.emitFeedsLike({
   uid: this.data.uid,
   fid: this.data.fid,
   likes,
   hasLike
  })
  })
 },
 ...
 }
})

这样,我们便在小程序中实现了一套跨页面数据同步的方案。

直观上这已经非常完美的实现了我们的需求。但在小程序中存在一个与我们常规经验不太一致的地方。那就是页面在关掉后,它里面的对象并没有销毁,这点是因为小程序的逻辑层是共用一个进程。

详解小程序开发经验:多页面数据同步

因此,每次进入页面,都会注册一次监听事件,而退出页面后,该事件并不会销毁。这样的话,多次重复进入页面,就会注册多个重复事件,当事件发生时,就会执行多次响应。请仔细观察下图!

详解小程序开发经验:多页面数据同步

为了避免该现象出现,我们切记要在页面的onUnload事件中,主动销毁监听事件。

Page({
 eventsListener: {},
 data: {
 list: []
 },
 onLoad: function (options) {
 ...
 // 监听点赞事件广播
 ↓ 重点在这里 ↓
 this.eventsListener.feedsLike = App.events.on('feedsLike', data => {
  console.log('我的动态页面收到点赞变化通知:', data)
  // 进行更新操作
 })
 // 监听发布事件广播
 ↓ 重点在这里 ↓
 this.eventsListener.publishFeeds= App.events.on('publishFeeds', data => {
  console.log('我的动态页面收到发布动态通知:', data)
  // 进行更新操作
 })
 },
 ↓ 重点在这里 ↓
 onUnload() {
 for (let i in this.eventsListener) {
  App.events.remove(i, this.eventsListener[i])
 } 
 },
 ...
})

至此,我们在小程序中完美的实现了跨页面/组件、多页面数据同步。

本文研究的demo均可以小程序中体验,项目源码:git.weixin.qq.com/xinyuanliu/…

以上所述是小编给大家介绍的小程序开发经验:多页面数据同步详解整合,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对三水点靠木网站的支持!

Javascript 相关文章推荐
JavaScript 基础问答三
Dec 03 Javascript
javascript学习之闭包分析
Dec 02 Javascript
formvalidator验证插件中有关ajax验证问题
Jan 04 Javascript
jquery win 7透明弹出层效果的简单代码
Aug 06 Javascript
angularjs中的单元测试实例
Dec 06 Javascript
Jquery判断radio、selelct、checkbox是否选中及获取选中值方法总结
Apr 15 Javascript
Vue 2.0在IE11中打开项目页面空白的问题解决
Jul 16 Javascript
AngularJS实现表单验证功能详解
Oct 12 Javascript
vue使用axios跨域请求数据问题详解
Oct 18 Javascript
React如何避免重渲染
Apr 10 Javascript
vue2 中二级路由高亮问题及配置方法
Jun 10 Javascript
Vue.js 中的实用工具方法【推荐】
Jul 04 Javascript
JavaScript实现星级评价效果
May 17 #Javascript
JavaScript实现美化滑块效果
May 17 #Javascript
vue中使用mxgraph的方法实例代码详解
May 17 #Javascript
vue中引入mxGraph的步骤详解
May 17 #Javascript
微信小程序云开发 搭建一个管理小程序
May 17 #Javascript
微信小程序云开发实现增删改查功能
May 17 #Javascript
微信小程序云开发实现云数据库读写权限
May 17 #Javascript
You might like
php chr() ord()中文截取乱码问题解决方法
2008/09/08 PHP
Fatal error: session_start(): Failed to initialize storage module: files问题解决方法
2014/05/04 PHP
php模拟post上传图片实现代码
2016/06/24 PHP
php字符串比较函数用法小结(strcmp,strcasecmp,strnatcmp及strnatcasecmp)
2016/07/18 PHP
js获取网页高度(详细整理)
2012/12/28 Javascript
jQuery表单域选择器用法分析
2015/02/10 Javascript
Angularjs制作简单的路由功能demo
2015/04/14 Javascript
轻松学习jQuery插件EasyUI EasyUI实现树形网络基本操作(2)
2015/11/30 Javascript
Jquery1.9.1源码分析系列(十五)动画处理之外篇
2015/12/04 Javascript
jQuery实现简易的输入框字数计数功能示例
2017/01/16 Javascript
VUE2.0+Element-UI+Echarts封装的组件实例
2018/03/02 Javascript
解决vue打包项目后刷新404的问题
2018/03/06 Javascript
JavaScript实现表单注册、表单验证、运算符功能
2018/10/15 Javascript
angular 实现同步验证器跨字段验证的方法
2019/04/11 Javascript
详解一些适用于Node.js的命名约定
2019/12/08 Javascript
[59:42]Secret vs Alliacne 2019国际邀请赛小组赛 BO2 第一场 8.15
2019/08/17 DOTA
Python抓取淘宝下拉框关键词的方法
2015/07/08 Python
简单讲解Python中的字符串与字符串的输入输出
2016/03/13 Python
python中hashlib模块用法示例
2017/10/30 Python
python按行读取文件,去掉每行的换行符\n的实例
2018/04/19 Python
Django 使用logging打印日志的实例
2018/04/28 Python
Python实现的列表排序、反转操作示例
2019/03/13 Python
Python获取好友地区分布及好友性别分布情况代码详解
2019/07/10 Python
python运用sklearn实现KNN分类算法
2019/10/16 Python
来自美国主售篮球鞋的零售商店:KICKSUSA
2017/11/28 全球购物
Erwin Müller穆勒家居瑞士官网:您整个家庭的邮购公司
2019/12/28 全球购物
奢华时尚的创新平台:Baltini
2020/10/03 全球购物
公务员培训自我鉴定
2013/09/19 职场文书
小学语文国培感言
2014/03/04 职场文书
2014年绩效考核工作总结
2014/12/11 职场文书
企业爱心捐款倡议书
2015/04/27 职场文书
小学教师师德师风承诺书
2015/04/28 职场文书
军训结束新闻稿
2015/07/17 职场文书
phpQuery解析HTML乱码问题(补充官网未列出的乱码解决方案)
2021/04/01 PHP
面试分析分布式架构Redis热点key大Value解决方案
2022/03/13 Redis
MySql中的json_extract函数处理json字段详情
2022/06/05 MySQL