【Android Jetpack高手日志】LiveData 从入门到精通

}

在 NameActivity 中使用

class NameActivity : AppCompatActivity() {

// Use the ‘by viewModels()’ Kotlin property delegate

// from the activity-ktx artifact

private val model: NameViewModel by viewModels()

override fun onCreate(savedInstanceState: Bundle?) {

super.onCreate(savedInstanceState)

// Other code to setup the activity…

// Create the observer which updates the UI.

val nameObserver = Observer { newName ->

// Update the UI, in this case, a TextView.

nameTextView.text = newName

}

// Observe the LiveData, passing in this activity as the LifecycleOwner and the observer.

//观察数据

model.currentName.observe(this, nameObserver)

}

}

更新数据

button.setOnClickListener {

val anotherName = “John Doe”

model.currentName.setValue(anotherName)

}

您必须调用setValue(T)方法以从主线程更新 LiveData 对象。如果非主线程中执行代码,您可以改用 postValue(T)方法来更新 LiveData 对象。

将 LiveData 与 Room 一起使用

Room 持久性库支持返回 LiveData 对象的可观察查询。可观察查询属于数据库访问对象 (DAO) 的一部分。

当数据库更新时,Room 会生成更新 LiveData 对象所需的所有代码。在需要时,生成的代码会在后台线程上异步运行查询。此模式有助于使界面中显示的数据与存储在数据库中的数据保持同步。您可以在 Room 持久性库指南中详细了解 Room 和 DAO。

将协程与 LiveData 一起使用

LiveData 支持 Kotlin 协程。如需了解详情,请参阅将 Kotlin 协程与 Android 架构组件一起使用

其实 Google 推出的这一系列 Android Jetpack 库,在不同库之间的配合已经做的非常好了,比如上面的 LiveData 和 Room,协程一起使用,配合起来如丝般顺滑,非常好用。

进阶使用
扩展 LiveData

如果观察者的生命周期处于 STARTEDRESUMED 状态,则 LiveData 会认为该观察者处于活跃状态。以下示例代码说明了如何扩展 LiveData 类:

class StockLiveData(symbol: String) : LiveData() {

private val stockManager = StockManager(symbol)

private val listener = { price: BigDecimal ->

value = price

}

override fun onActive() {

stockManager.requestPriceUpdates(listener)

}

override fun onInactive() {

stockManager.removeUpdates(listener)

}

}

  • LiveData 对象具有活跃观察者时,会调用 onActive() 方法。这意味着,您需要从此方法开始观察股价更新。

  • LiveData 对象没有任何活跃观察者时,会调用 onInactive() 方法。由于没有观察者在监听,因此没有理由与 StockManager 服务保持连接。

  • setValue(T) 方法将更新 LiveData 实例的值,并将更改告知活跃观察者。

public class MyFragment : Fragment() {

override fun onViewCreated(view: View, savedInstanceState: Bundle?) {

super.onViewCreated(view, savedInstanceState)

val myPriceListener: LiveData = …

myPriceListener.observe(viewLifecycleOwner, Observer { price: BigDecimal? ->

// Update the UI.

})

}

}

observe() 方法将与 Fragment 视图关联的 LifecycleOwner 作为第一个参数传递。这样做表示此观察者已绑定到与所有者关联的 Lifecycle 对象,这意味着:

  • 如果 Lifecycle 对象未处于活跃状态,那么即使值发生更改,也不会调用观察者

  • 销毁 Lifecycle 对象后,会自动移除观察者。

LiveData具有生命周期感知能力,也就是说可以在多个 Activity、Fragment、Service 之间共享这些对象,所以可以将其设置为单例

转换 LiveData

如果你希望将 LiveData 分派给观察者之前对其值进行修改,或者说另一个 LiveData 需要根据它做进一步处理,Lifecycle 包中的 Transformation 类可以为我们提供帮助:

Transformations.map()

对存储在 LiveData 对象中的值应用函数,并将结果传播到下游。

val userLiveData: LiveData = UserLiveData()

val userName: LiveData = Transformations.map(userLiveData) {

user -> “${user.name} ${user.lastName}”

}

Transformations.switchMap()

如果想要根据某个值 切换观察不同LiveData数据,则可以使用Transformations.switchMap()方法。

//两个liveData,由liveDataSwitch决定 返回哪个livaData数据

MutableLiveData liveData3 = new MutableLiveData<>();

MutableLiveData liveData4 = new MutableLiveData<>();

//切换条件LiveData,liveDataSwitch的value 是切换条件

MutableLiveData liveDataSwitch = new MutableLiveData<>();

//liveDataSwitchMap由switchMap()方法生成,用于添加观察者

LiveData liveDataSwitchMap = Transformations.switchMap(liveDataSwitch, new Function<Boolean, LiveData>() {

@Override

public LiveData apply(Boolean input) {

//这里是具体切换逻辑:根据liveDataSwitch的value返回哪个liveData

if (input) {

return liveData3;

}

return liveData4;

}

});

liveDataSwitchMap.observe(this, new Observer() {

@Override

public void onChanged(String s) {

Log.i(TAG, "onChanged2: " + s);

}

});

boolean switchValue = true;

liveDataSwitch.setValue(switchValue);//设置切换条件值

liveData3.setValue(“liveData3”);

liveData4.setValue(“liveData4”);

这个官方的 demo 没看明白,引用的是胡飞洋大佬博客文章内容,上面的很多用例都是官方的,因为在我看来官方的用例是最好的。

合并多个 LiveData 源

MediatorLiveDataLiveData 的子类,允许您合并多个 LiveData 源。只要任何原始的 LiveData 源对象发生更改,就会触发 MediatorLiveData 对象的观察者。

比如一个常见的功能,app 内的通知常用小红点显示是否有新消息,这个时候,如果没有对红点的展示逻辑做一个统一的抽象和管理的话,就会感觉很复杂,后续也不太好维护。这个时候就可以需要MediatorLiveData进行对红点的统一管理,详细内容可以参考这篇文章

源码分析

我们知道LiveData是在Lifecycle 的帮助下,实现了生命周期管理的一致性。对于 LiveData 的数据修改流程,我们从setValue方法开始分析。

@MainThread

protected void setValue(T value) {

assertMainThread(“setValue”);

mVersion++;

mData = value;

dispatchingValue(null);

}

然后进入 dispatchingValue方法

@SuppressWarnings(“WeakerAccess”) /* synthetic access */

void dispatchingValue(@Nullable ObserverWrapper initiator) {

if (mDispatchingValue) {

mDispatchInvalidated = true;

return;

}

mDispatchingValue = true;

do {

mDispatchInvalidated = false;

if (initiator != null) {

considerNotify(initiator);

initiator = null;

} else {

for (Iterator<Map.Entry<Observer<? super T>, ObserverWrapper>> iterator =

mObservers.iteratorWithAdditions(); iterator.hasNext(); ) {

considerNotify(iterator.next().getValue());

if (mDispatchInvalidated) {

break;

}

}

}

} while (mDispatchInvalidated);

mDispatchingValue = false;

}

因为传入的dispatchingValue的参数为 null,所以我们进入considerNotify方法

private void considerNotify(ObserverWrapper observer) {

if (!observer.mActive) {

return;

}

// Check latest state b4 dispatch. Maybe it changed state but we didn’t get the event yet.

//

// we still first check observer.active to keep it as the entrance for events. So even if

// the observer moved to an active state, if we’ve not received that event, we better not

// notify for a more predictable notification order.

if (!observer.shouldBeActive()) {

observer.activeStateChanged(false);

return;

}

if (observer.mLastVersion >= mVersion) {

return;

}

observer.mLastVersion = mVersion;

observer.mObserver.onChanged((T) mData);

}

可以看到在最后一行,被观察者通知观察者进行onChanged数据变更,而我们的订阅是在 observe方法中

@MainThread

public void observe(@NonNull LifecycleOwner owner, @NonNull Observer<? super T> observer) {

assertMainThread(“observe”);

//如果 LifecycleOwner 的 DESTROYED,return

if (owner.getLifecycle().getCurrentState() == DESTROYED) {

// ignore

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()的真正实现是 Activity 中的 LifecycleRegistry`(如果在Activity 中调用的话)。

owner.getLifecycle().addObserver(wrapper);

}

可以看到整个流程就是这么简单。就是简单的观察者模式的运用。不过 LiveData 还帮助我们自动处理的生命周期,防止内存泄漏。

要注意的是, LiveData#dispatchingValue除了在我们主动更新数据的时候会触发, 在我们的观察者状态变更(inactive->active)的时候, 也会通知到, 这就导致了LiveData必然支持粘性事件

setValue/postValue方法,是为了将数据通知到主线程,其中 postValue方法最终调用了setValue

protected void postValue(T value) {

boolean postTask;

synchronized (mDataLock) {

postTask = mPendingData == NOT_SET;

mPendingData = value;

}

if (!postTask) {

return;

}

ArchTaskExecutor.getInstance().postToMainThread(mPostValueRunnable); //发送到主线程

}

mPostValueRunnable 实例的内容如下:

private final Runnable mPostValueRunnable = new Runnable() {

@SuppressWarnings(“unchecked”)

@Override

public void run() {

Object newValue;

synchronized (mDataLock) {

newValue = mPendingData;

mPendingData = NOT_SET;

}

setValue((T) newValue); //最终还是调用了 setValue方法

}

};

使用 LiveData 的优势

确保界面符合数据状态

LiveData 遵循观察者模式。当底层数据发生变化时,LiveData 会通知 Observer 对象。您可以整合代码以在这些 Observer 对象中更新界面。这样一来,您无需在每次应用数据发生变化时更新界面,因为观察者会替您完成更新。

不会发生内存泄漏

观察者会绑定到 Lifecycle 对象,并在其关联的生命周期遭到销毁后进行自我清理。

不会因 Activity 停止而导致崩溃

如果观察者的生命周期处于非活跃状态(如返回栈中的 Activity),则它不会接收任何 LiveData 事件。

不再需要手动处理生命周期

界面组件只是观察相关数据,不会停止或恢复观察。LiveData 将自动管理所有这些操作,因为它在观察时可以感知相关的生命周期状态变化。

数据始终保持最新状态

如果生命周期变为非活跃状态,它会在再次变为活跃状态时接收最新的数据。例如,曾经在后台的 Activity 会在返回前台后立即接收最新的数据。

适当的配置更改

如果由于配置更改(如设备旋转)而重新创建了 Activity 或 Fragment,它会立即接收最新的可用数据。

共享资源

您可以使用单例模式扩展 LiveData 对象以封装系统服务,以便在应用中共享它们。LiveData 对象连接到系统服务一次,然后需要相应资源的任何观察者只需观察 LiveData 对象。如需了解详情,请参阅扩展 LiveData

基于 LiveData 的事件总线的实现(LiveDataBus)

可以看出, LiveData本身就已经可观测数据更新, 我们通过维护一张 eventName-LiveData 的哈希表, 就可以得到一个基础的事件总线

class LiveDataBus {

internal val liveDatas by lazy { mutableMapOf<String, LiveData<*>>() }

@Synchronized

private fun bus(channel: String): LiveData{

return liveDatas.getOrPut(channel){

LiveDataEvent(channel)

} as LiveData

}

fun with(channel: String): LiveData{

return bus(channel)

}

companion object{

private val INSTANCE by lazy { LiveDataBus() }

@JvmStatic

fun get() = INSTANCE

}

}

除了粘性事件以外,我们还需要非粘性事件的支持, 这里有两种做法。反射获取ObserverWrapper.mLastVersion, 在订阅的时候使得初始化的ObserverWrapper.mLastVersion等于LiveData.mVersion, 使得粘性消息无法通过实现;这里可以参考美团的文章:[Android消息总线的演进之路:用LiveDataBus替代RxBus、EventBus]。

还有一种做法是粘性消息最终会调到Observer#onChanged, 那么我们就干脆将其再进行一层包装, 内部维护实际的订阅消息数, 来判断是否要触发真正的onChanged方法,详细内容可以参考这篇文章

LiveData 的建议写法

在我自己的项目中,我通常使用该方式来定义LiveData,严格控制LiveData的权限,确保视图 Activity/Fragment 中只有该数据的访问权限,而将数据的修改权限严格控制在ViewModel中,参考自 github 上的一些优秀项目的写法。

//ViewModel.java

private val _cartCountLD: MutableLiveData = MutableLiveData()

val cartCountLD: LiveData

get() = _cartCountLD

private val _saveBirthdayLD: MutableLiveData = MutableLiveData()

val saveBirthdayLD: LiveData

get() = _saveBirthdayLD

灵魂拷问(在 Activity 的哪个几个生命周期修改 LiveData 值,能被观察者观察到)

前面说过数据变更的通知控制在生命周期活跃状态 STARTED、RESUMED,那么对应是哪几个生命周期呢?先来看看这段代码:

class MainActivity : AppCompatActivity() {

private var jetpackLiveData: MutableLiveData = MutableLiveData()

private val TAG = “================”

override fun onCreate(savedInstanceState: Bundle?) {

super.onCreate(savedInstanceState)

setContentView(R.layout.activity_main)

jetpackLiveData.observe(this,Observer {

Log.i(TAG, “observe value:”+it+" state:"+this.lifecycle.currentState)

})

Log.i(TAG, “onCreate: “+” state:”+this.lifecycle.currentState)

jetpackLiveData.value = “onCreate”

}

override fun onStart() {

super.onStart()

Log.i(TAG, “onStart: “+” state:”+this.lifecycle.currentState)

jetpackLiveData.value = “onStart”

}

override fun onResume() {

super.onResume()

Log.i(TAG, “onResume: “+” state:”+this.lifecycle.currentState)

jetpackLiveData.value = “onResume”

}

override fun onPause() {

super.onPause()

Log.i(TAG, “onPause: “+” state:”+this.lifecycle.currentState)

jetpackLiveData.value = “onPause”

}

override fun onStop() {

super.onStop()

Log.i(TAG, “onStop: “+” state:”+this.lifecycle.currentState)

jetpackLiveData.value = “onStop”

}

override fun onDestroy() {

super.onDestroy()

Log.i(TAG, “onDestroy: “+” state:”+this.lifecycle.currentState)

jetpackLiveData.value = “onDestroy”

}

}

启动界面并退出界面,观察到的日志是:

jetpackdemo I/================: onCreate: state:INITIALIZED

jetpackdemo I/================: onStart: state:CREATED

jetpackdemo I/================: observe value:onStart state:STARTED //STARTED 状态

总结

最后为了帮助大家深刻理解Android相关知识点的原理以及面试相关知识,这里放上相关的我搜集整理的24套腾讯、字节跳动、阿里、百度2019-2021面试真题解析,我把技术点整理成了视频和PDF(实际上比预期多花了不少精力),包知识脉络 + 诸多细节

还有 高级架构技术进阶脑图、Android开发面试专题资料 帮助大家学习提升进阶,也节省大家在网上搜索资料的时间来学习,也可以分享给身边好友一起学习。

一线互联网面试专题

379页的Android进阶知识大全

379页的Android进阶知识大全

网上学习 Android的资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。希望这份系统化的技术体系对大家有一个方向参考。

2021年虽然路途坎坷,都在说Android要没落,但是,不要慌,做自己的计划,学自己的习,竞争无处不在,每个行业都是如此。相信自己,没有做不到的,只有想不到的。祝大家2021年万事大吉。
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》点击传送门,即可获取!

启动界面并退出界面,观察到的日志是:

jetpackdemo I/================: onCreate: state:INITIALIZED

jetpackdemo I/================: onStart: state:CREATED

jetpackdemo I/================: observe value:onStart state:STARTED //STARTED 状态

总结

最后为了帮助大家深刻理解Android相关知识点的原理以及面试相关知识,这里放上相关的我搜集整理的24套腾讯、字节跳动、阿里、百度2019-2021面试真题解析,我把技术点整理成了视频和PDF(实际上比预期多花了不少精力),包知识脉络 + 诸多细节

还有 高级架构技术进阶脑图、Android开发面试专题资料 帮助大家学习提升进阶,也节省大家在网上搜索资料的时间来学习,也可以分享给身边好友一起学习。

[外链图片转存中…(img-NENfYuqa-1715155783071)]

[外链图片转存中…(img-xubng6nx-1715155783072)]

[外链图片转存中…(img-z8OEJsSE-1715155783072)]

网上学习 Android的资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。希望这份系统化的技术体系对大家有一个方向参考。

2021年虽然路途坎坷,都在说Android要没落,但是,不要慌,做自己的计划,学自己的习,竞争无处不在,每个行业都是如此。相信自己,没有做不到的,只有想不到的。祝大家2021年万事大吉。
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》点击传送门,即可获取!

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值