热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

深入解析JetpackLiveData的工作原理

本文详细介绍了Jetpack库中的LiveData组件,这是一种能够感知生命周期并可被观察的数据持有类。LiveData支持存储任何类型的数据,并通常与ViewModel结合使用,以确保数据在配置变更时保持一致。

概述:LiveData 是 Jetpack 库的一部分,设计用于持有可被观察的数据,这些数据能够响应组件(如 Activity 或 Fragment)的生命周期变化。LiveData 不限制数据类型,支持任何数据类型的存储,常与 ViewModel 一起使用,以实现数据在不同配置下的持久性和一致性。观察者需在主线程中绑定 LiveData,数据更新的通知也将在主线程中处理。



LiveData 的优势:


1. 避免内存泄漏:当关联的组件被销毁时,LiveData 会自动清理相关联的观察者。
2. 自动管理生命周期:只有当关联组件处于活跃状态时,LiveData 才会触发数据更新,减少了不必要的资源消耗。
3. 降低组件间的耦合度:数据与界面逻辑分离,提高了代码的可维护性。
4. 即时更新:组件状态变化时,LiveData 能迅速做出反应,提供最新的数据。
5. 恢复时自动更新:当组件从非活跃状态恢复至活跃状态时,LiveData 会自动推送最新数据给观察者。



源码解析:


为了更好地理解 LiveData 的工作机制,我们将通过源码对其进行分析。首先,在项目中引入最新的 LiveData 和 ViewModel 库:


dependencies {
implementation 'androidx.lifecycle:lifecycle-livedata:2.3.1'
implementation 'androidx.lifecycle:lifecycle-viewmodel:2.3.1'
}

接下来,我们创建一个 ViewModel 类,并在其中实例化 LiveData 对象:



然后,在 Activity 中调用 LiveData 的 observe 方法,注册观察者:



在 observe 方法中,LiveData 通过 LifecycleBoundObserver 将观察者与组件关联起来,并将其保存在内部的映射表中。如果组件的生命周期已结束,则不会添加观察者,避免不必要的资源占用。


public void observe(@NonNull LifecycleOwner owner, @NonNull Observer observer) {
assertMainThread("observe");
if (owner.getLifecycle().getCurrentState() == DESTROYED) {
return;
}
LifecycleBoundObserver wrapper = new LifecycleBoundObserver(owner, observer);
ObserverWrapper existing = mObservers.putIfAbsent(observer, wrapper);
if (existing != null && !existing.isAttachedTo(owner)) {
throw new IllegalArgumentException("Cannot add the same observer with different lifecycles");
}
if (existing != null) {
return;
}
owner.getLifecycle().addObserver(wrapper);
}

当组件的生命周期状态发生变化时,LifecycleBoundObserver 的 onStateChanged 方法会被调用,根据组件的状态决定是否通知观察者更新数据。如果组件被销毁,观察者将从 LiveData 中移除,防止内存泄漏。


public void onStateChanged(@NonNull LifecycleOwner source, @NonNull Lifecycle.Event event) {
if (mOwner.getLifecycle().getCurrentState() == DESTROYED) {
removeObserver(mObserver);
return;
}
activeStateChanged(shouldBeActive());
}

通过调用 setValue 或 postValue 方法,可以更新 LiveData 中的数据。setValue 必须在主线程调用,而 postValue 可以在后台线程调用,它会将任务发布到主线程队列中,最终调用 setValue 方法。


@MainThread
protected void setValue(T value) {
assertMainThread("setValue");
mVersion++;
mData = value;
dispatchingValue(null);
}

protected void postValue(T value) {
boolean postTask;
synchronized (mDataLock) {
postTask = mPendingData == NOT_SET;
mPendingData = value;
}
if (!postTask) {
return;
}
ArchTaskExecutor.getInstance().postToMainThread(mPostValueRunnable);
}

dispatchingValue 方法负责遍历所有活跃的观察者,并调用 considerNotify 方法来决定是否需要通知观察者。considerNotify 方法检查观察者的活跃状态,并在必要时调用观察者的 onChanged 方法,传递最新的数据值。


private void considerNotify(ObserverWrapper observer) {
if (!observer.mActive) {
return;
}
if (!observer.shouldBeActive()) {
observer.activeStateChanged(false);
return;
}
if (observer.mLastVersion >= mVersion) {
return;
}
observer.mLastVersion = mVersion;
observer.mObserver.onChanged((T) mData);
}

通过上述过程,LiveData 确保了数据更新的及时性和准确性,同时有效地管理了组件的生命周期,提高了应用的稳定性和性能。


推荐阅读
author-avatar
喜怒哀乐168_572
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有