2024年最全HarmonyOS 应用开发之自定义组件冻结功能,面试真题解析小区快递点乱收费,整改后记者来了解

img
img

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化的资料的朋友,可以戳这里获取

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

    Tabs() {
      TabContent() {
        FreezeChild({ message: this.message })
      }.tabBar('one')

      TabContent() {
        FreezeChild({ message: this.message })
      }.tabBar('two')
    }
  }
  .width('100%')
}
.height('100%')

}
}

@Component({ freezeWhenInactive: true })
struct FreezeChild {
@Link @Watch(“onMessageUpdated”) message: number
private index: number = 0

onMessageUpdated() {
console.info(FreezeChild message callback func ${this.message}, index: ${this.index})
}

build() {
Text(“message” + ${this.message}, index: ${this.index})
.fontSize(50)
.fontWeight(FontWeight.Bold)
}
}


在上面的示例中:


1.点击“change message”更改message的值,当前正在显示的TabContent组件中的@Watch中注册的方法onMessageUpdated被触发。


2.点击“two”切换到另外的TabContent,TabContent状态由inactive变为active,对应的@Watch中注册的方法onMessageUpdated被触发。


3.再次点击“change message”更改message的值,仅当前显示的TabContent子组件中的@Watch中注册的方法onMessageUpdated被触发。


![](https://img-blog.csdnimg.cn/img_convert/8d0ca70eee611491b841a3c4c34c23e5.gif)


#### LazyforEach


* 对LazyforEach中缓存的自定义组件进行冻结,不会触发组件的更新。



// Basic implementation of IDataSource to handle data listener
class BasicDataSource implements IDataSource {
private listeners: DataChangeListener[] = [];
private originDataArray: string[] = [];

public totalCount(): number {
return 0;
}

public getData(index: number): string {
return this.originDataArray[index];
}

// 该方法为框架侧调用,为LazyForEach组件向其数据源处添加listener监听
registerDataChangeListener(listener: DataChangeListener): void {
if (this.listeners.indexOf(listener) < 0) {
console.info(‘add listener’);
this.listeners.push(listener);
}
}

// 该方法为框架侧调用,为对应的LazyForEach组件在数据源处去除listener监听
unregisterDataChangeListener(listener: DataChangeListener): void {
const pos = this.listeners.indexOf(listener);
if (pos >= 0) {
console.info(‘remove listener’);
this.listeners.splice(pos, 1);
}
}

// 通知LazyForEach组件需要重载所有子组件
notifyDataReload(): void {
this.listeners.forEach(listener => {
listener.onDataReloaded();
})
}

// 通知LazyForEach组件需要在index对应索引处添加子组件
notifyDataAdd(index: number): void {
this.listeners.forEach(listener => {
listener.onDataAdd(index);
})
}

// 通知LazyForEach组件在index对应索引处数据有变化,需要重建该子组件
notifyDataChange(index: number): void {
this.listeners.forEach(listener => {
listener.onDataChange(index);
})
}

// 通知LazyForEach组件需要在index对应索引处删除该子组件
notifyDataDelete(index: number): void {
this.listeners.forEach(listener => {
listener.onDataDelete(index);
})
}
}

class MyDataSource extends BasicDataSource {
private dataArray: string[] = [];

public totalCount(): number {
return this.dataArray.length;
}

public getData(index: number): string {
return this.dataArray[index];
}

public addData(index: number, data: string): void {
this.dataArray.splice(index, 0, data);
this.notifyDataAdd(index);
}

public pushData(data: string): void {
this.dataArray.push(data);
this.notifyDataAdd(this.dataArray.length - 1);
}
}

@Entry
@Component
struct LforEachTest {
private data: MyDataSource = new MyDataSource();
@State @Watch(“onMessageUpdated”) message: number = 0;

onMessageUpdated() {
console.info(LazyforEach message callback func ${this.message})
}

aboutToAppear() {
for (let i = 0; i <= 20; i++) {
this.data.pushData(Hello ${i})
}
}

build() {
Column() {
Button(‘change message’).onClick(() => {
this.message++
})
List({ space: 3 }) {
LazyForEach(this.data, (item: string) => {
ListItem() {
FreezeChild({ message: this.message, index: item })
}
}, (item: string) => item)
}.cachedCount(5).height(500)
}

}
}

@Component({ freezeWhenInactive: true })
struct FreezeChild {
@Link @Watch(“onMessageUpdated”) message: number;
private index: string = “”;

aboutToAppear() {
console.info(FreezeChild aboutToAppear index: ${this.index})
}

onMessageUpdated() {
console.info(FreezeChild message callback func ${this.message}, index: ${this.index})
}

build() {
Text(“message” + ${this.message}, index: ${this.index})
.width(‘90%’)
.height(160)
.backgroundColor(0xAFEEEE)
.textAlign(TextAlign.Center)
.fontSize(30)
.fontWeight(FontWeight.Bold)
}
}


在上面的示例中:


1.点击“change message”更改message的值,当前正在显示的ListItem中的子组件@Watch中注册的方法onMessageUpdated被触发。缓存节点@Watch中注册的方法不会被触发。(如果不加组件冻结,当前正在显示的ListItem和cachcount缓存节点@Watch中注册的方法onMessageUpdated都会触发watch回调。)


2.List区域外的ListItem滑动到List区域内,状态由inactive变为active,对应的@Watch中注册的方法onMessageUpdated被触发。


3.再次点击“change message”更改message的值,仅有当前显示的ListItem中的子组件@Watch中注册的方法onMessageUpdated被触发。


![](https://img-blog.csdnimg.cn/img_convert/f5617fef12c6c952ae1c8a65aca0a011.gif)


#### Navigation


* 对当前不可见的页面进行冻结,不会触发组件的更新,当返回该页面时,触发@Watch回调进行刷新。



@Entry
@Component
struct MyNavigationTestStack {
@Provide(‘pageInfo’) pageInfo: NavPathStack = new NavPathStack();
@State @Watch(“info”) message: number = 0;
@State logNumber: number = 0;

info() {
console.info(freeze-test MyNavigation message callback ${this.message});
}

@Builder
PageMap(name: string) {
if (name === ‘pageOne’) {
pageOneStack({ message: this.message, logNumber: this.logNumber })
} else if (name === ‘pageTwo’) {
pageTwoStack({ message: this.message, logNumber: this.logNumber })
} else if (name === ‘pageThree’) {
pageThreeStack({ message: this.message, logNumber: this.logNumber })
}
}

build() {
Column() {
Button(‘change message’)
.onClick(() => {
this.message++;
})
Navigation(this.pageInfo) {
Column() {
Button(‘Next Page’, { stateEffect: true, type: ButtonType.Capsule })
.width(‘80%’)
.height(40)
.margin(20)
.onClick(() => {
this.pageInfo.pushPath({ name: ‘pageOne’ }); //将name指定的NavDestination页面信息入栈
})
}
}.title(‘NavIndex’)
.navDestination(this.PageMap)
.mode(NavigationMode.Stack)
}
}
}

@Component
struct pageOneStack {
@Consume(‘pageInfo’) pageInfo: NavPathStack;
@State index: number = 1;
@Link message: number;
@Link logNumber: number;

build() {
NavDestination() {
Column() {
NavigationContentMsgStack({ message: this.message, index: this.index, logNumber: this.logNumber })
Text(“cur stack size:” + ${this.pageInfo.size()})
.fontSize(30)
.fontWeight(FontWeight.Bold)
Button(‘Next Page’, { stateEffect: true, type: ButtonType.Capsule })
.width(‘80%’)
.height(40)
.margin(20)
.onClick(() => {
this.pageInfo.pushPathByName(‘pageTwo’, null);
})
Button(‘Back Page’, { stateEffect: true, type: ButtonType.Capsule })
.width(‘80%’)
.height(40)
.margin(20)
.onClick(() => {
this.pageInfo.pop();
})
}.width(‘100%’).height(‘100%’)
}.title(‘pageOne’)
.onBackPressed(() => {
this.pageInfo.pop();
return true;
})
}
}

@Component
struct pageTwoStack {
@Consume(‘pageInfo’) pageInfo: NavPathStack;
@State index: number = 2;
@Link message: number;
@Link logNumber: number;

build() {
NavDestination() {
Column() {
NavigationContentMsgStack({ message: this.message, index: this.index, logNumber: this.logNumber })
Text(“cur stack size:” + ${this.pageInfo.size()})
.fontSize(30)
.fontWeight(FontWeight.Bold)
Button(‘Next Page’, { stateEffect: true, type: ButtonType.Capsule })
.width(‘80%’)
.height(40)
.margin(20)
.onClick(() => {
this.pageInfo.pushPathByName(‘pageThree’, null);
})
Button(‘Back Page’, { stateEffect: true, type: ButtonType.Capsule })
.width(‘80%’)
.height(40)
.margin(20)
.onClick(() => {
this.pageInfo.pop();
})
}.width(‘100%’).height(‘100%’)
}.title(‘pageTwo’)
.onBackPressed(() => {
this.pageInfo.pop();
return true;
})
}
}

@Component
struct pageThreeStack {
@Consume(‘pageInfo’) pageInfo: NavPathStack;
@State index: number = 3;
@Link message: number;
@Link logNumber: number;

build() {
NavDestination() {
Column() {
NavigationContentMsgStack({ message: this.message, index: this.index, logNumber: this.logNumber })
Text(“cur stack size:” + ${this.pageInfo.size()})
.fontSize(30)
.fontWeight(FontWeight.Bold)
Button(‘Next Page’, { stateEffect: true, type: ButtonType.Capsule })
.width(‘80%’)
.height(40)
.margin(20)
.onClick(() => {
this.pageInfo.pushPathByName(‘pageOne’, null);
})
Button(‘Back Page’, { stateEffect: true, type: ButtonType.Capsule })
.width(‘80%’)
.height(40)
.margin(20)
.onClick(() => {
this.pageInfo.pop();
})
}.width(‘100%’).height(‘100%’)
}.title(‘pageThree’)
.onBackPressed(() => {
this.pageInfo.pop();
return true;
})
}
}

@Component({ freezeWhenInactive: true })
struct NavigationContentMsgStack {
@Link @Watch(“info”) message: number;
@Link index: number;
@Link logNumber: number;

info() {
console.info(freeze-test NavigationContent message callback ${this.message});
console.info(freeze-test ---- called by content ${this.index});
this.logNumber++;
}

build() {
Column() {
Text(“msg:” + ${this.message})
.fontSize(30)
.fontWeight(FontWeight.Bold)
Text(“log number:” + ${this.logNumber})
.fontSize(30)

深知大多数程序员,想要提升技能,往往是自己摸索成长,但自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!


img
img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上鸿蒙开发知识点,真正体系化!

由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新

需要这份系统化的资料的朋友,可以戳这里获取

摸索成长,但自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!**


[外链图片转存中…(img-CcCrOrIq-1715652649405)]
[外链图片转存中…(img-RZoc7077-1715652649405)]

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上鸿蒙开发知识点,真正体系化!

由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新

需要这份系统化的资料的朋友,可以戳这里获取

  • 4
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值