4.4【HarmonyOS 鸿蒙开发】组件 ListContainer(下) 性能优化
作者:韩茹
公司:程序咖(北京)科技有限公司
鸿蒙巴士专栏作家
在适配 ListContainer 的数据时,无论是新创建的列表项实例,还是从缓存中获取到的,都须要调用办法 findComponentById() 获取所有子组件并进行数据填充,大量调用该办法,会损耗 ListContainer 的性能。比拟好的解决方案是在创立列表项实例时进行调用,将获取到的所有子组件绑定到列表项的实例中,当从缓存中获取到列表项实例后,间接应用绑定的的子组件填充新数据。
咱们从新写一个例子,预期效果图如下:
<img src=”https://img.chengxuka.com/listcontaineryunxing9.gif” alt=”listcontaineryunxing9″ style=”zoom:50%;” />
这里阐明一下,文档接上一篇,我的项目我也没有从新创立,还是在上一篇文档中的我的项目根底上来写的。
残缺示例代码如下:
1、首先咱们先在 layout 下新建一个布局文件。listcontainer_layout2.xml,咱们先搁置一个 ListContainer 控件:
<?xml version="1.0" encoding="utf-8"?>
<DirectionalLayout
xmlns:ohos="http://schemas.huawei.com/res/ohos"
ohos:height="match_parent"
ohos:width="match_parent"
ohos:padding="8vp"
ohos:orientation="vertical">
<ListContainer
ohos:id="$+id:list_container3"
ohos:height="match_parent"
ohos:width="match_parent"/>
</DirectionalLayout>
而后持续创立 xml 文件,用于示意 ListContainer 中每个 Item 的布局款式,list_item_demo3_setting.xml,代码如下:
<?xml version="1.0" encoding="utf-8"?>
<DirectionalLayout
xmlns:ohos="http://schemas.huawei.com/res/ohos"
ohos:height="80vp"
ohos:width="match_parent"
ohos:padding="8vp"
ohos:orientation="horizontal">
<Image
ohos:id="$+id:ima_setting"
ohos:height="match_parent"
ohos:width="0"
ohos:layout_alignment="vertical_center"
ohos:weight="2">
</Image>
<Text
ohos:id="$+id:text_setting"
ohos:height="match_content"
ohos:width="0"
ohos:padding="4fp"
ohos:text_size="20fp"
ohos:start_padding="8vp"
ohos:end_padding="8vp"
ohos:weight="6"
ohos:layout_alignment="vertical_center"/>
<Switch
ohos:id="$+id:switch_setting"
ohos:height="20vp"
ohos:width="0vp"
ohos:weight="1"
ohos:layout_alignment="vertical_center"/>
</DirectionalLayout>
2、graphic 目录下,咱们创立好资源文件,因为这里用到了 switch 开关按钮,咱们设计几个款式:
thumb_off_element.xml:
<?xml version="1.0" encoding="utf-8"?>
<shape xmlns:ohos="http://schemas.huawei.com/res/ohos"
ohos:shape="oval">
<solid
ohos:color="#FFFFFF"/>
<bounds
ohos:top="0"
ohos:left="0"
ohos:right="20vp"
ohos:bottom="20vp"/>
</shape>
thumb_on_element.xml:
<?xml version="1.0" encoding="utf-8"?>
<shape xmlns:ohos="http://schemas.huawei.com/res/ohos"
ohos:shape="oval">
<solid
ohos:color="#1E90FF"/>
<bounds
ohos:top="0"
ohos:left="0"
ohos:right="20vp"
ohos:bottom="20vp"/>
</shape>
track_off_element.xml:
<?xml version="1.0" encoding="utf-8"?>
<shape xmlns:ohos="http://schemas.huawei.com/res/ohos"
ohos:shape="rectangle">
<solid
ohos:color="#808080"/>
<corners
ohos:radius="20vp"/>
</shape>
track_on_element.xml:
<?xml version="1.0" encoding="utf-8"?>
<shape xmlns:ohos="http://schemas.huawei.com/res/ohos"
ohos:shape="rectangle">
<solid
ohos:color="#87CEFA"/>
<corners
ohos:radius="20vp"/>
</shape>
到此对于页面布局局部,咱们曾经筹备结束。
3、在 provider 包下,新建一个文件 SettingItem.java。该类中的属性,就是对应 ListContainer 中每个 Item 条目对应的数据。
package com.example.hanrulistcontainer.provider;
public class SettingItem {
private int imageId;
private String settingName;
private boolean isChecked;
public SettingItem(int imageId, String settingName, boolean isChecked) {
this.imageId = imageId;
this.settingName = settingName;
this.isChecked = isChecked;
}
public int getImageId() {return imageId;}
public void setImageId(int imageId) {this.imageId = imageId;}
public String getSettingName() {return settingName;}
public void setSettingName(String settingName) {this.settingName = settingName;}
public boolean isChecked() {return isChecked;}
public void setChecked(boolean checked) {isChecked = checked;}
}
4、在该包下持续创立 java 文件,创立 SettingProvider.java
package com.example.hanrulistcontainer.provider;
import com.example.hanrulistcontainer.ResourceTable;
import ohos.aafwk.ability.AbilitySlice;
import ohos.agp.components.*;
import ohos.agp.components.element.ShapeElement;
import ohos.agp.components.element.StateElement;
import java.util.List;
public class SettingProvider extends BaseItemProvider {
// ListContainer 的数据汇合
private List<SettingItem> settingList;
private AbilitySlice slice;
public SettingProvider(List<SettingItem> list, AbilitySlice slice) {
this.settingList = list;
this.slice = slice;
}
@Override
public int getCount() {return settingList == null ? 0 : settingList.size();
}
@Override
public Object getItem(int position) {if (settingList != null && position >= 0 && position < settingList.size()){return settingList.get(position);
}
return null;
}
@Override
public long getItemId(int position) {return position;}
@Override
public Component getComponent(int position, Component component, ComponentContainer componentContainer) {
final Component cpt;
SettingHolder holder;
SettingItem setting = settingList.get(position);
if (component == null) {cpt = LayoutScatter.getInstance(slice).parse(ResourceTable.Layout_list_item_demo3_setting, null, false);
holder = new SettingHolder(cpt);
// 将获取到的子组件信息绑定到列表项的实例中
cpt.setTag(holder);
} else {
cpt = component;
// 从缓存中获取到列表项实例后,间接应用绑定的子组件信息进行数据填充。holder = (SettingHolder) cpt.getTag();}
holder.settingIma.setPixelMap(setting.getImageId());
holder.settingText.setText(setting.getSettingName());
holder.settingSwitch.setChecked(setting.isChecked());
return cpt;
}
// 用于保留列表项中的子组件信息
// 应用该外部类,目标是为了缩小 findComponentById 被调用的次数,从而优化性能。public class SettingHolder {
Image settingIma;
Text settingText;
Switch settingSwitch;
SettingHolder(Component component) {settingIma = (Image) component.findComponentById(ResourceTable.Id_ima_setting);
settingText = (Text) component.findComponentById(ResourceTable.Id_text_setting);
settingSwitch = (Switch) component.findComponentById(ResourceTable.Id_switch_setting);
// 设置 switch 的开关
settingSwitch.setTrackElement(trackElementInit(new ShapeElement(slice, ResourceTable.Graphic_track_on_element),
new ShapeElement(slice, ResourceTable.Graphic_track_off_element)));
settingSwitch.setThumbElement(thumbElementInit(new ShapeElement(slice, ResourceTable.Graphic_thumb_on_element),
new ShapeElement(slice, ResourceTable.Graphic_thumb_off_element)));
}
private StateElement trackElementInit(ShapeElement on, ShapeElement off) {StateElement trackElement = new StateElement();
trackElement.addState(new int[]{ComponentState.COMPONENT_STATE_CHECKED}, on);
trackElement.addState(new int[]{ComponentState.COMPONENT_STATE_EMPTY}, off);
return trackElement;
}
private StateElement thumbElementInit(ShapeElement on, ShapeElement off) {StateElement thumbElement = new StateElement();
thumbElement.addState(new int[]{ComponentState.COMPONENT_STATE_CHECKED}, on);
thumbElement.addState(new int[]{ComponentState.COMPONENT_STATE_EMPTY}, off);
return thumbElement;
}
}
}
阐明:这里应用了 SettingHolder 这个外部类,目标是为了缩小 findComponentById 被调用的次数。
5、在 slice 目录下创立一个新的 AbilitySlice 文件:ThirdAbilitySlice.java
package com.example.hanrulistcontainer.slice;
import com.example.hanrulistcontainer.ResourceTable;
import com.example.hanrulistcontainer.provider.SettingItem;
import com.example.hanrulistcontainer.provider.SettingProvider;
import ohos.aafwk.ability.AbilitySlice;
import ohos.aafwk.content.Intent;
import ohos.agp.components.ListContainer;
import java.util.ArrayList;
import java.util.List;
public class ThirdAbilitySlice extends AbilitySlice {
@Override
protected void onStart(Intent intent) {super.onStart(intent);
setUIContent(ResourceTable.Layout_listcontainer_layout2);
ListContainer listContainer = (ListContainer) findComponentById(ResourceTable.Id_list_container3);
SettingProvider provider = new SettingProvider(getData(), this);
listContainer.setItemProvider(provider);
}
private List<SettingItem> getData() {ArrayList<SettingItem> data = new ArrayList<>();
for (int i = 0; i < 100; i++) {
data.add(new SettingItem(
ResourceTable.Media_icon,
"SettingName" + i,
i % 3 == 0
));
}
return data;
}
}
6、批改一下程序的 入口:
package com.example.hanrulistcontainer;
import com.example.hanrulistcontainer.slice.MainAbilitySlice;
import com.example.hanrulistcontainer.slice.SecondAbilitySlice;
import com.example.hanrulistcontainer.slice.ThirdAbilitySlice;
import ohos.aafwk.ability.Ability;
import ohos.aafwk.content.Intent;
public class MainAbility extends Ability {
@Override
public void onStart(Intent intent) {super.onStart(intent);
// super.setMainRoute(MainAbilitySlice.class.getName());
// super.setMainRoute(SecondAbilitySlice.class.getName());
super.setMainRoute(ThirdAbilitySlice.class.getName());
}
}
到此代码局部完结,运行程序即可。
目前为止,咱们写的数据都是假数据,等到咱们学习了网络,就能够从网络上获取数据,而后解析,通过 ListContainer 展现进去了。
更多内容:
1、社区:鸿蒙巴士 https://www.harmonybus.net/
2、公众号:HarmonyBus
3、技术交换 QQ 群:714518656
4、视频课:https://www.chengxuka.com