欢迎来到258分享网,纯净的网络源码分享基地!

258资源分享网

全部作品
全部作品
网站源码
微信源码
素材特效
源码插件
视频教程
建站学院
热门搜索: 织梦  农业种植  农业  安全设置  官方
258资源分享 > 建站学院 > 微信开发 > 小程序redux性能优化,提升三倍渲染速度

推荐下载

HTML5响应式自适应网咯设计

2020-05-12   浏览:739

高端HTML5响应式企业通用网

2020-05-06   浏览:521

html5响应式外贸网站英文版

2020-05-08   浏览:510

HTML5自适应律师工作室类网

2020-04-04   浏览:503

HTML5影视传媒文化公司类网

2020-05-12   浏览:498

小程序redux性能优化,提升三倍渲染速度

发布时间:2020-10-13  
前言

最近用户反馈我们的小程序很卡,打开商品列表需要四五秒时间,带着这个疑问,我决定对小程序做个全面的性能优化,要做性能优化,必须先理清以下三个关键点。

1、产生性能问题的关键点 
2、度量性能指标 
3、寻找解决方案

在阅读案例分析前,建议能先了解小程序的工作原理和性能关键点。

 

工作原理 (官方说明)

小程序的视图层目前使用 WebView 作为渲染载体,而逻辑层是由独立的 JavascriptCore 作为运行环境。在架构上,WebView 和 JavascriptCore 都是独立的模块,并不具备数据直接共享的通道。当前,视图层和逻辑层的数据传输,实际上通过两边提供的 evaluateJavascript 所实现。即用户传输的数据,需要将其转换为字符串形式传递,同时把转换后的数据内容拼接成一份 JS 脚本,再通过执行 JS 脚本的形式传递到两边独立环境。 
而 evaluateJavascript 的执行会受很多方面的影响,数据到达视图层并不是实时的。

 

性能关键点(官方说明)

频繁的去 setData 
在我们分析过的一些案例里,部分小程序会非常频繁(毫秒级)的去setData,其导致了两个后果:

Android 下用户在滑动时会感觉到卡顿,操作反馈延迟严重,因为 JS 线程一直在编译执行渲染,未能及时将用户操作事件传递到逻辑层,逻辑层亦无法及时将操作处理结果及时传递到视图层; 
渲染有出现延时,由于 WebView 的 JS 线程一直处于忙碌状态,逻辑层到页面层的通信耗时上升,视图层收到的数据消息时距离发出时间已经过去了几百毫秒,渲染的结果并不实时;

每次 setData 都传递大量新数据 
由setData的底层实现可知,我们的数据传输实际是一次 evaluateJavascript 脚本过程,当数据量过大时会增加脚本的编译执行时间,占用 WebView JS 线程,

后台态页面进行 setData 
当页面进入后台态(用户不可见),不应该继续去进行setData,后台态页面的渲染用户是无法感受的,另外后台态页面去setData也会抢占前台页面的执行。

 

度量性能指标

我们在优化性能时,指标是非常重要的,没有指标,你没法知道优化的点是否有效。不能单凭感觉去优化,要根据指标反馈,明确优化的成果。同时,优化就像个无底洞,要注意投入产出比。 
用户反馈的卡顿,要么就是js执行消耗资源过多导致处理器没响应,要么是UI渲染消耗资源过多,导致UI没法响应用户操作。 
通过查看代码,我们并没有消耗大量计算资源的业务逻辑,但是出现了UI反复操作和抢占资源的现象。

 

如何度量

可以利用setData的第二个参数,传入callback函数,统计渲染时长。代码如下

 

let startTime = Date.now()

this.setData(data, () => {

let endTime = Data.now()

console.log(endTime - startTime, '渲染时长')

})

案例分析  1、检查点:是否频繁去setData  检查结果:存在  产生原因:redux中监听的是整个store,只要store变化,就会执行setData操作,这就意味着页面无关的数据改变,也会触发该页面执行setData操作,但是这个操作是无意义的。  问题代码:

 

// libs/redux-wechat/connect.js

 

// 对整个store进行subscribe。变化就执行handleChange

this.unsubscribe = this.store.subscribe(handleChange.bind(this, options));

 

function handleChange(options) {

...省略代码

const state = this.store.getState()

const mappedState = mapState(state, options);

this.setData(mappedState)

}

解决方案:

只监听当前页面用到的store中的部分数据,只有该部分数据变化,才setData。(store没提供单个数据的监听,如果自己修改redux实现,难度较大,同时修改太底层,容易出不可预料的异常。)  判断页面数据与需要更新数据是否相同,如果相同,不做操作。(这个方案成本比较低,就用它吧)

代码实现:

 

// libs/redux-wechat/connect.js

// 如果更新的数据和页面数据相同,不做操作。

function handleChange(options) {

...省略代码

const state = this.store.getState()

const mappedState = mapState(state, options);

// 如果更新的数据和页面数据相同,不做操作。

if (mappedState === this.prevState) return // 新加入代码

this.setData(mappedState)

// 保存上一次数据

this.prevState = mappedState // 新加入代码