Android ListView列表優(yōu)化的方法詳解
前言
列表 ListView 是應(yīng)用中最為常見(jiàn)的組件,而列表往往也會(huì)承載很多元素,當(dāng)元素多,尤其是那種圖片文件比較大的場(chǎng)合,就可能會(huì)導(dǎo)致列表卡頓,嚴(yán)重的時(shí)候可能導(dǎo)致應(yīng)用崩潰。本篇來(lái)介紹如何優(yōu)化列表。
優(yōu)化點(diǎn)1:使用 builder構(gòu)建列表
當(dāng)你的列表元素是動(dòng)態(tài)增長(zhǎng)的時(shí)候(比如上拉加載更多),請(qǐng)不要直接用children 的方式,一直往children 的數(shù)組增加組件,那樣會(huì)很糟糕。
//糟糕的用法 ListView( ??children:?[ ????item1, ????item2, ????item3, ????... ??], ) //正確的用法 ListView.builder( ??itemBuilder:?(context,?index)?=>?ListItem(), ??itemCount:?itemCount, )
對(duì)于 ListView.builder 是按需構(gòu)建列表元素,也就是只有那些可見(jiàn)的元素才會(huì)調(diào)用itemBuilder 構(gòu)建元素,這樣對(duì)于大列表而言性能開(kāi)銷自然會(huì)小很多。
Creates a scrollable, linear array of widgets that are created on demand. This constructor is appropriate for list views with a large (or infinite) number of children because the builder is called only for those children that are actually visible.
優(yōu)化點(diǎn)2:禁用 addAutomaticKeepAlives 和 addRepaintBoundaries 特性
這兩個(gè)屬性都是為了優(yōu)化滾動(dòng)過(guò)程中的用戶體驗(yàn)的。addAutomaticKeepAlives 特性默認(rèn)是 true,意思是在列表元素不可見(jiàn)后可以保持元素的狀態(tài),從而在再次出現(xiàn)在屏幕的時(shí)候能夠快速構(gòu)建。這其實(shí)是一個(gè)拿空間換時(shí)間的方法,會(huì)造成一定程度的內(nèi)存開(kāi)銷??梢栽O(shè)置為 false 關(guān)閉這一特性。缺點(diǎn)是滑動(dòng)過(guò)快的時(shí)候可能會(huì)出現(xiàn)短暫的白屏(實(shí)際會(huì)很少發(fā)生)。
addRepaintBoundaries 是將列表元素使用一個(gè)重繪邊界(Repaint Boundary)包裹,從而使得滾動(dòng)的時(shí)候可以避免重繪。而如果列表很容易繪制(列表元素布局比較簡(jiǎn)單的情況下)的時(shí)候,可以關(guān)閉這個(gè)特性來(lái)提高滾動(dòng)的流暢度。
addAutomaticKeepAlives:?false, addRepaintBoundaries:?false,
優(yōu)化點(diǎn)3:盡可能將列表元素中不變的組件使用 const 修飾
使用 const 相當(dāng)于將元素緩存起來(lái)實(shí)現(xiàn)共用,若列表元素某些部分一直保持不變,那么可以使用 const 修飾。
return?Padding(
??child:?Row(
????children:?[
??????const?ListImage(),
??????const?SizedBox(
????????width:?5.0,
??????),
??????Text('第$index?個(gè)元素'),
????],
??),
??padding:?EdgeInsets.all(10.0),
);
優(yōu)化點(diǎn)4:使用 itemExtent 確定列表元素滾動(dòng)方向的尺寸
對(duì)于很多列表,我們?cè)跐L動(dòng)方向上的尺寸是提前可以根據(jù) UI設(shè)計(jì)稿知道的,如果能夠知道的話,那么使用 itemExtent 屬性制定列表元素在滾動(dòng)方向的尺寸,可以提升性能。這是因?yàn)椋绻恢付ǖ脑?,在滾動(dòng)過(guò)程中,會(huì)需要推算每個(gè)元素在滾動(dòng)方向的尺寸從而消耗計(jì)算資源。
itemExtent:?120,
優(yōu)化實(shí)例
下面是一開(kāi)始未改造的列表,嗯,可以認(rèn)為是垃圾代碼。
class?LargeListView?extends?StatefulWidget?{
??const?LargeListView({Key??key})?:?super(key:?key);
??@override
??_LargeListViewState?createState()?=>?_LargeListViewState();
}
class?_LargeListViewState?extends?State<LargeListView>?{
??@override
??Widget?build(BuildContext?context)?{
????return?Scaffold(
??????appBar:?AppBar(
????????title:?Text('大列表'),
????????brightness:?Brightness.dark,
??????),
??????body:?ListView(
????????children:?List.generate(
??????????1000,
??????????(index)?=>?Padding(
????????????padding:?EdgeInsets.all(10.0),
????????????child:?Row(
??????????????children:?[
????????????????Image.network(
??????????????????'https://p1-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/7869eac08a7d4177b600dc7d64998204~tplv-k3u1fbpfcp-watermark.jpeg',
??????????????????width:?200,
????????????????),
????????????????const?SizedBox(
??????????????????width:?5.0,
????????????????),
????????????????Text('第$index?個(gè)元素'),
??????????????],
????????????),
??????????),
????????),
??????),
????);
??}
}
當(dāng)然,實(shí)際不會(huì)是用 List.generate 來(lái)生成列表元素,但是也不要用一個(gè) List<Widget> 列表對(duì)象一直往里面加列表元素,然后把這個(gè)列表作為 ListView 的 children!改造后的代碼如下所示,因?yàn)閷⒘斜碓夭鸱值酶?xì),代碼量是多一些,但是性能上會(huì)好很多。
import?'package:flutter/material.dart';
class?LargeListView?extends?StatefulWidget?{
??const?LargeListView({Key??key})?:?super(key:?key);
??@override
??_LargeListViewState?createState()?=>?_LargeListViewState();
}
class?_LargeListViewState?extends?State<LargeListView>?{
??@override
??Widget?build(BuildContext?context)?{
????return?Scaffold(
??????appBar:?AppBar(
????????title:?Text('大列表'),
????????brightness:?Brightness.dark,
??????),
??????body:?ListView.builder(
????????itemBuilder:?(context,?index)?=>?ListItem(
??????????index:?index,
????????),
????????itemCount:?1000,
????????addAutomaticKeepAlives:?false,
????????addRepaintBoundaries:?false,
????????itemExtent:?120.0,
??????),
????);
??}
}
class?ListItem?extends?StatelessWidget?{
??final?int?index;
??ListItem({Key??key,?required?this.index})?:?super(key:?key);
??@override
??Widget?build(BuildContext?context)?{
????return?Padding(
??????child:?Row(
????????children:?[
??????????const?ListImage(),
??????????const?SizedBox(
????????????width:?5.0,
??????????),
??????????Text('第$index?個(gè)元素'),
????????],
??????),
??????padding:?EdgeInsets.all(10.0),
????);
??}
}
class?ListImage?extends?StatelessWidget?{
??const?ListImage({Key??key})?:?super(key:?key);
??@override
??Widget?build(BuildContext?context)?{
????return?Image.network(
??????'https://p1-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/7869eac08a7d4177b600dc7d64998204~tplv-k3u1fbpfcp-watermark.jpeg',
??????width:?200,
????);
??}
}總結(jié)
本篇介紹了 Flutter ListView 的4個(gè)優(yōu)化要點(diǎn),非常實(shí)用哦!實(shí)際上,這些要點(diǎn)都可以從官網(wǎng)的文檔里找出對(duì)應(yīng)的說(shuō)明。因此,如果遇到了性能問(wèn)題,除了搜索引擎外,也建議多看看官方的文檔。另外一個(gè),對(duì)于列表圖片,有時(shí)候也需要前后端配合,比如目前的手機(jī)都是號(hào)稱1億像素的,如果上傳的時(shí)候直接上傳原圖,那么加載如此大的圖片肯定是非常消耗資源的。對(duì)于這種情況,建議是生成列表縮略圖(可能需要針對(duì)不同屏幕尺寸生成不同的縮略圖)。
以上就是Android ListView列表優(yōu)化的方法詳解的詳細(xì)內(nèi)容,更多關(guān)于Android ListView列表優(yōu)化的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Android使用GRPC進(jìn)行通信過(guò)程解析
這篇文章主要給大家介紹了在Android上使用grpc的方法教程,文中通過(guò)示例代碼給大家詳細(xì)介紹了在android上使用grpc的方法以及可能遇到的種種問(wèn)題的解決方法,對(duì)大家具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來(lái)一起看看吧2023-02-02
Android開(kāi)發(fā)筆記之:返回鍵的復(fù)寫(xiě)onBackPressed()介紹
本篇文章是對(duì)Android中返回鍵的復(fù)寫(xiě)onBackPressed()進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下2013-05-05
揭秘雙十一手機(jī)淘寶圖標(biāo)如何被動(dòng)態(tài)更換
這篇文章主要介紹了每到雙十一十二的時(shí)候Android手機(jī)動(dòng)態(tài)更換手機(jī)圖標(biāo)的方法,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-08-08
實(shí)例講解Android app開(kāi)發(fā)中ListView的基本使用及優(yōu)化
這篇文章主要介紹了Android app開(kāi)發(fā)中ListView的基本使用及優(yōu)化,ListView視圖組件是Android中最常用的組件之一需要的朋友可以參考下2016-02-02
android自定義gradle插件并且發(fā)布到本地倉(cāng)庫(kù)詳細(xì)教程
這篇文章主要介紹了android自定義gradle插件并且發(fā)布到本地倉(cāng)庫(kù)詳細(xì)教程的相關(guān)資料,需要的朋友可以參考下2023-07-07
Android實(shí)現(xiàn)系統(tǒng)打印功能
這篇文章主要為大家詳細(xì)介紹了Android實(shí)現(xiàn)系統(tǒng)打印功能,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-12-12

