為您解碼網(wǎng)站建設(shè)的點(diǎn)點(diǎn)滴滴
發(fā)表日期:2019-11 文章編輯:小燈 瀏覽次數(shù):4338
在公司小程序也開發(fā)了一段時(shí)間了,中間遇到過(guò)很多問(wèn)題,特此記錄幾個(gè)比較典型的問(wèn)題和解決方案。
此問(wèn)題提供源碼demo,可導(dǎo)入微信開發(fā)者工具查看。
textarea 是小程序的原生組件,它的一個(gè)表現(xiàn)就是優(yōu)先級(jí)很高,這導(dǎo)致了一些困擾,比如我們有一個(gè)表單頁(yè)面,最下面就是一個(gè)textarea和一個(gè)保存按鈕,這會(huì)導(dǎo)致textarea的文字會(huì)浮現(xiàn)在按鈕上。如下圖:
它最大的問(wèn)題時(shí)會(huì)導(dǎo)致保存按鈕可能點(diǎn)擊無(wú)效或者會(huì)彈出鍵盤,并且開發(fā)者工具模擬器和真機(jī)表現(xiàn)不一樣,這真是個(gè)坑!
模擬器中,針對(duì) position:fixed
定位的按鈕,我們加一個(gè) z-index:10
即可, z-index
等于多少合適不清楚,試了等于1是不行的,10就可以,其余的值沒(méi)試過(guò)。
.submit-cls {
position: fixed;
left: 30px;
right: 30px;
bottom: 300px;
text-align: center;
background-color: green;
color: #fff;
z-index: 10;
}
模擬器中的表現(xiàn):
然兒,真機(jī)上(Android)依然無(wú)效!如下圖:
于是我想到了 cover-view 標(biāo)簽,cover-view 是微信提供的一個(gè)原生組件,它是覆蓋在原生組件之上的文本視圖,可覆蓋的原生組件包括 map、video、canvas、camera、live-player、live-pusher之上,只支持嵌套 cover-view、cover-image,可在 cover-view 中使用 button。
用 cover-view 標(biāo)簽包裹 button 如何呢?郁悶的事情發(fā)生了,真機(jī)上按鈕不見(jiàn)了!。。。這方法貌似不行。。
<cover-view>
<button class="submit-cls" id='button' bindtap="onClick"> Button z-index: 10 </button>
</cover-view>
那我直接用 cover-view 標(biāo)簽作為按鈕呢?
<cover-view class="cover-view-clas" id='cover-view' bindtap="onClick"> cover-view z-index: 10 </cover-view>
.cover-view-clas {
position: fixed;
height: 40px;
line-height: 40px;
left: 30px;
right: 30px;
bottom: 250px;
text-align: center;
background-color: orangered;
color: #fff;
}
結(jié)果在模擬器里不行
但是真機(jī)上表現(xiàn)很好。于是我也加了一個(gè) z-index: 10
,這樣模擬器和真機(jī)表現(xiàn)就一致。
綜上所述,要解決這個(gè)問(wèn)題似乎只有一個(gè)辦法,那就是用 cover-view
+ z-index:10
,然兒這樣會(huì)導(dǎo)致一個(gè)的副作用,沒(méi)法使用微信的開放能力比如 open-type
。
我們知道,與傳統(tǒng)的瀏覽器Web頁(yè)面最大區(qū)別在于,小程序的是基于 雙線程 模型的,在這種架構(gòu)中,小程序的渲染層使用 WebView 作為渲染載體,而邏輯層則由獨(dú)立的 JsCore 線程運(yùn)行 JS 腳本,雙方并不具備數(shù)據(jù)直接共享的通道,因此渲染層和邏輯層的通信要由 Native 的 JSBrigde 做中轉(zhuǎn)。
每當(dāng)小程序視圖數(shù)據(jù)需要更新時(shí),邏輯層會(huì)調(diào)用小程序宿主環(huán)境提供的 setData
方法將數(shù)據(jù)從邏輯層傳遞到視圖層,經(jīng)過(guò)一系列渲染步驟之后完成UI視圖更新。然而當(dāng) setData
傳遞大量的新數(shù)據(jù)、頻繁的執(zhí)行 setData
操作、過(guò)多的頁(yè)面節(jié)點(diǎn)數(shù)時(shí)會(huì)影響渲染性能。
意思是, setData 只用來(lái)通知頁(yè)面更新,只有需要通知頁(yè)面更新的時(shí)候,頁(yè)面引用了某個(gè) data 字段時(shí)才使用,其它字段數(shù)據(jù)我們有時(shí)候可能只是為了讓 js 方便使用。比如如下數(shù)據(jù)
data: {
form: {
name: 'xxxx',
... ...
},
index: 0
}
假如 頁(yè)面上根本沒(méi)用到 index 來(lái)展示,只是我們的邏輯變量,那么我們?cè)谫x值的時(shí)候就直接 this.data.index = xxx
即可,不要用 setData 去賦值了。
setData 是支持使用 數(shù)據(jù)路徑 的方式對(duì)對(duì)象的局部字段進(jìn)行更新,我們可能會(huì)遇到這樣的場(chǎng)景: list 列表是從后臺(tái)獲取的數(shù)據(jù),并展示在頁(yè)面上,當(dāng) list 列表的第一項(xiàng)數(shù)據(jù)的 src 字段需要更新時(shí),一般情況下我們會(huì)從后臺(tái)獲取新的 list 列表,執(zhí)行 setData 更新整個(gè) list 列表。
// 后臺(tái)獲取列表數(shù)據(jù)
const list = requestSync();
// 更新整個(gè)列表
this.setData({ list });
實(shí)際上,只有個(gè)別字段需要更新時(shí),我們可以這么寫來(lái)避免整個(gè) list 列表更新:
// 后臺(tái)獲取列表數(shù)據(jù)
const list = requestSync();
// 局部更新列表
this.setData({
'list[0].src': list[0].src
});
小程序自定義組件的實(shí)現(xiàn)是由小程序官方設(shè)計(jì)的 Exparser 框架所支持,框架實(shí)現(xiàn)的自定義組件的組件模型與 Web Components 標(biāo)準(zhǔn)的 Shadow DOM 相似:
在頁(yè)面引用自定義組件后,當(dāng)初始化頁(yè)面時(shí),Exparser 會(huì)在創(chuàng)建頁(yè)面實(shí)例的同時(shí),也會(huì)根據(jù)自定義組件的注冊(cè)信息進(jìn)行組件實(shí)例化,然后根據(jù)組件自帶的 data 數(shù)據(jù)和組件WXML,構(gòu)造出獨(dú)立的 Shadow Tree ,并追加到頁(yè)面 Composed Tree 。創(chuàng)建出來(lái)的 Shadow Tree 擁有著自己獨(dú)立的邏輯空間、數(shù)據(jù)、樣式環(huán)境及setData調(diào)用,這是組件化帶來(lái)的好處。
基于自定義組件的 Shadow DOM 模型設(shè)計(jì),我們可以將頁(yè)面中一些需要高頻執(zhí)行 setData 更新的功能模塊(如倒計(jì)時(shí)、進(jìn)度條等)封裝成自定義組件嵌入到頁(yè)面中。當(dāng)這些自定義組件視圖需要更新時(shí),執(zhí)行的是組件自己的 setData ,新舊節(jié)點(diǎn)樹的對(duì)比計(jì)算和渲染樹的更新都只限于組件內(nèi)有限的節(jié)點(diǎn)數(shù)量,有效降低渲染時(shí)間開銷。
在項(xiàng)目中,有一個(gè)預(yù)約模塊,字段忒多,保險(xiǎn)業(yè)務(wù)嘛,需要用戶填寫各種數(shù)據(jù)的,為了用戶體驗(yàn)拆成了多個(gè)步驟,如圖
一開始,業(yè)務(wù)上要求切換tab的時(shí)候數(shù)據(jù)要緩存,跟Vue的 keep-alive 一樣,但是小程序沒(méi)有這樣的機(jī)制,所以利用小程序的 hidden
屬性,也就是 Vue 中的 v-show
,組件始終會(huì)被渲染,只是簡(jiǎn)單的控制顯示與隱藏。關(guān)于wx:if 和 hidden。
這樣的導(dǎo)致頁(yè)面節(jié)點(diǎn)太多,在低性能手機(jī)上會(huì)出現(xiàn)卡死的現(xiàn)象,直接無(wú)法渲染或者渲染太慢。
后來(lái)改為 wx:if
來(lái)切換
<view wx:if="{{current === 0}}">......</view>
<view wx:if="{{current === 1}}">......</view>
<view wx:if="{{current === 2}}">......</view>
... ...
這樣以來(lái)一次渲染節(jié)點(diǎn)太多的問(wèn)題解決了,但是怎么實(shí)現(xiàn)tab切換的時(shí)候輸入的內(nèi)容杯緩存呢?其實(shí)我們的笨辦法就是切換的時(shí)候把前一個(gè)表單內(nèi)容保存到 localStorage 或 gloabData 中,切換回去的時(shí)候再取出來(lái)填充,這中間會(huì)有一個(gè)明顯的渲染過(guò)程,肉眼可見(jiàn),沒(méi)辦法,目前只能犧牲一點(diǎn)點(diǎn)體驗(yàn)了。
對(duì)于這種大型表單,數(shù)據(jù)處理和邏輯交互的時(shí)候要非常注意,很容易出現(xiàn)性能問(wèn)題。
這次就說(shuō)這么多吧,文章如有什么錯(cuò)誤,或有什么想法,請(qǐng)留言,不吝賜教!
日期:2019-11 瀏覽次數(shù):5554
日期:2019-11 瀏覽次數(shù):12008
日期:2019-11 瀏覽次數(shù):4379
日期:2019-11 瀏覽次數(shù):5419
日期:2019-11 瀏覽次數(shù):5280
日期:2019-11 瀏覽次數(shù):7221
日期:2019-11 瀏覽次數(shù):5185
日期:2019-11 瀏覽次數(shù):15796
日期:2019-11 瀏覽次數(shù):4757
日期:2019-11 瀏覽次數(shù):6544
日期:2019-11 瀏覽次數(shù):5406
日期:2019-11 瀏覽次數(shù):4594
日期:2019-11 瀏覽次數(shù):10797
日期:2019-11 瀏覽次數(shù):8351
日期:2019-11 瀏覽次數(shù):5105
日期:2019-11 瀏覽次數(shù):4338
日期:2019-11 瀏覽次數(shù):8991
日期:2019-11 瀏覽次數(shù):4680
日期:2019-11 瀏覽次數(shù):4869
日期:2019-11 瀏覽次數(shù):4887
日期:2019-11 瀏覽次數(shù):4507
日期:2019-11 瀏覽次數(shù):5056
日期:2019-11 瀏覽次數(shù):10311
日期:2019-11 瀏覽次數(shù):5485
日期:2019-11 瀏覽次數(shù):5466
日期:2019-11 瀏覽次數(shù):4921
日期:2019-11 瀏覽次數(shù):12366
日期:2019-11 瀏覽次數(shù):7380
日期:2019-11 瀏覽次數(shù):7931
日期:2019-11 瀏覽次數(shù):4875
Copyright ? 2013-2018 Tadeng NetWork Technology Co., LTD. All Rights Reserved.