经验首页 前端设计 程序设计 Java相关 移动开发 数据库/运维 软件/图像 大数据/云计算 其他经验
当前位置:技术经验 » 移动开发 » 微信小程序 » 查看文章
浅谈小程序 setData学问多
来源:jb51  时间:2019/2/21 9:23:06  对本文有异议

为什么不能频繁 setData

先科普下 setData 做的事情:

在数据传输时,逻辑层会执行一次 JSON.stringify 来去除掉 setData 数据中不可传输的部分,之后将数据发送给视图层。同时,逻辑层还会将 setData 所设置的数据字段与 data 合并,使开发者可以用 this.data 读取到变更后的数据。

因此频繁调用,视图会一直更新,阻塞用户交互,引发性能问题。

但频繁调用是常见开发场景,能不能频繁调用的同时,视图延迟更新呢?

参考 Vue,我们能知道,Vue 每次赋值操作并不会直接更新视图,而是缓存到一个数据更新队列中,异步更新,再触发渲染,此时多次赋值,也只会渲染一次。

于是有网友就给出了这套方案的实现方法:

  1. let newState = null
  2. const asyncSetData = ({
  3. vm,
  4. newData,
  5. }) => {
  6. newState = {
  7. ...newState,
  8. ...newData,
  9. }
  10. Promise.resolve().then(() => {
  11. if (!newState) return
  12. vm.setData({
  13. ...newState,
  14. })
  15. newState = null
  16. })
  17. }

由于异步代码会在同步代码之后执行,因此,当你多次使用 asyncSetData 设置 newState 时,newState 都会被缓存起来,并异步 setData 一次

但同时,这个方案也会带来一个新的问题,同步代码会阻塞页面的渲染。

同步代码会阻塞页面的渲染的问题其实在浏览器中也存在,但在小程序中,由于是逻辑、视图双线程架构,因此逻辑并不会阻塞视图渲染,这是小程序的优点,但在这套方案将会丢失这个优点。

鱼与熊掌不可兼得也!

对于信息流页面,数据过多怎么办

单次设置的数据不能超过 1024kB,请尽量避免一次设置过多的数据

通常,我们拉取到分页的数据 newList,添加到数组里,一般是这么写:

  1. this.setData({
  2. list: this.data.list.concat(newList)
  3. })

随着分页次数的增加,list 会逐渐增大,当超过 1024 kb 时,程序会报 exceed max data size 错误。

为了避免这个问题,我们可以直接修改 list 的某项数据,而不是对整个 list 重新赋值:

  1. let length = this.data.list.length;
  2. let newData = newList.reduce((acc, v, i)=>{
  3. acc[`list[${length+i}]`] = v;
  4. return acc;
  5. }, {});
  6. this.setData(newData);

这看着似乎还有点繁琐,为了简化操作,我们可以把 list 的数据结构从一维数组改为二维数组:list = [newList, newList], 每次分页,可以直接将整个 newList 赋值到 list 作为一个子数组,此时赋值方式为:

  1. let length = this.data.list.length;
  2. this.setData({
  3. [`list[${length}]`]: newList
  4. });

同时,模板也需要相应改成二重循环:

  1. <block wx:for="{{list}}" wx:for-item="{{listItem}}" wx:key="{{listItem}}">
  2. <child wx:for="{{listItem}}" wx:key="{{item}}"></child>
  3. </block>

下拉加载,让我们一夜回到解放前

信息流产品,总避免不了要做下拉加载。

下拉加载的数据,需要插到 list 的最前面,所以我们应该这样做:

  1. this.setData({
  2. 'list[-1]': newList
  3. })

哦不,对不起,上面是错的,应该是下面这样:

  1. this.setData({
  2. list: this.data.list.unshift(newList)
  3. });

这下好,又是一次性修改整个数组,一夜回到解放前......

为了解决这个问题,这里需要一点奇淫巧技:

  • 为下拉加载维护一个单独的二维数组 pullDownList
  • 在渲染时,用 wxs 将 pullDownList reverse 一下

此时,当下拉加载时,便可以只修改数组的某个子项:

  1. let length = this.data.pullDownList.length;
  2. this.setData({
  3. [`pullDownList[${length}]`]: newList
  4. });

关键在于渲染时候的反向渲染:

  1. <wxs module="utils">
  2. function reverseArr(arr) {
  3. console.log
  4. return arr.reverse()
  5. }
  6. module.exports = {
  7. reverseArr: reverseArr
  8. }
  9. </wxs>
  10. <block wx:for="{{utils.reverseArr(pullDownList)}}" wx:for-item="{{listItem}}" wx:key="{{listItem}}">
  11. <child wx:for="{{listItem}}" wx:key="{{item}}"></child>
  12. </block>
  13.  
  14. <block wx:for="{{list}}" wx:for-item="{{listItem}}" wx:key="{{listItem}}">
  15. <child wx:for="{{listItem}}" wx:key="{{item}}"></child>
  16. </block>

问题解决!以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持w3xue。

 友情链接:直通硅谷  点职佳  北美留学生论坛

本站QQ群:前端 618073944 | Java 606181507 | Python 626812652 | C/C++ 612253063 | 微信 634508462 | 苹果 692586424 | C#/.net 182808419 | PHP 305140648 | 运维 608723728

W3xue 的所有内容仅供测试,对任何法律问题及风险不承担任何责任。通过使用本站内容随之而来的风险与本站无关。
关于我们  |  意见建议  |  捐助我们  |  报错有奖  |  广告合作、友情链接(目前9元/月)请联系QQ:27243702 沸活量
皖ICP备17017327号-2 皖公网安备34020702000426号