LiveData

前言

LiveData 是一种可观察的数据存储器类。与常规的可观察类不同,LiveData 具有生命周期感知能力,意指它遵循其他应用组件(如 activityfragmentservice)的生命周期。这种感知能力可确保 LiveData 仅更新处于活跃生命周期状态的应用组件观察者。

说简单就是 LiveData 是一个可观察的数据存储类,内部借助了 Lifecycle,从而实现了生命周期感知,同时,这个可观察指的是,在其存储的数据更新时,它会去通知观察者。又因为生命周期感知的存在,所以可以做到 何时通知、何时解绑,从而做到安全无泄漏,就是如此:)

作者:Petterp
链接:https://juejin.cn/post/7173494700081414181
来源:稀土掘金
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

 特点

LiveData通常在ViewModel里面创建,这是因为以下原因:

1.确保系统不会从 Activity 或 Fragment 的 [onResume()]方法进行冗余调用。

2.确保 Activity 或 Fragment 变为活跃状态后具有可以立即显示的数据。一旦应用组件处于 STARTED 状态,就会从它正在观察的 LiveData 对象接收最新值。只有在设置了要观察的 LiveData 对象时,才会发生这种情况。

添加依赖

implementation 'androidx.lifecycle:lifecycle-livedata:2.4.0'
// or
implementation 'androidx.lifecycle:lifecycle-livedata-ktx:2.4.0'

初步使用示例

在ViewModel里创建LiveData

class NameViewModel : ViewModel() {

    // Create a LiveData with a String
    val mLiveData: MutableLiveData<String> by lazy {
        MutableLiveData<String>()
    }

    // Rest of the ViewModel...
}

在activity/fragment里观察

class NameActivity : AppCompatActivity() {

   private val  mViewModel: NameViewModel by viewModels()
  private lateinit var otcModelManager: OTCModelManager

 override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
  otcModelManager = SingleInstanceProvider.get(OTCModelManager::class.java)
    

 val nameObserver=Observer<String>{newName->
                 nameTextView.text = newName         
            }
   
    mViewModel.mLiveData.observe(this,nameObserver)

otcModelManager.getFaceDetectionQRCodeData(mViewModel.mLiveData)

}

}

class OTCModelManager()  {

  
 fun getFaceDetectionQRCodeData(mLiveData: MutableLiveData<String>){

    mLiveData.value = anotherName //该方式不可以在子线程使用
     //or
    mLiveData.postValue(anotherName) //该方式可以在子线程使用

}


}

Transformations

Lifecycle 软件包会提供 Transformations 类。该类会对LiveData进行一个辅助操作。其中主要会用到map(LiveData)switchMap(LiveData)

map(LiveData)的参数mapFunction(在kotlin里使用lambda表达式实现)的返回普通类型

 public static <X, Y> LiveData<Y> map(
             LiveData<X> source,
             final Function<X, Y> mapFunction) {...}

  使用案例: 

 val getFaceDetectionQRCodeSucLiveData:MutableLiveData<FaceDetectionQRCodeBean> 
by lazy {MutableLiveData<FaceDetectionQRCodeBean>() }

val stringLiveData  : LiveData<String> = Transformations.map(getFaceDetectionQRCodeSucLiveData) { bean ->
                "${bean.merchantName}"
            }

        stringLiveData.observe(owner=reference?.get()!!, onChanged = {
            LogUtil.e(TAG, "输出 stringLiveData 数据=${it}") //输出 stringLiveData 数据=ai全球鹰
        })

 

 switchMap(LiveData)的参数switchMapFunction(在kotlin里使用lambda表达式实现)返回LiveData类型。

 public static <X, Y> LiveData<Y> switchMap(
              LiveData<X> source,
             final Function<X, LiveData<Y>> switchMapFunction) {...}

   使用案例

 val getFaceDetectionQRCodeSucLiveData:MutableLiveData<FaceDetectionQRCodeBean>
 by lazy { MutableLiveData<FaceDetectionQRCodeBean>() }


  val getJGPushReceiverLiveData: MutableLiveData<String?> 
   by lazy{ MutableLiveData<String?>() }


 val resultLiveData: LiveData<String?> = Transformations.switchMap(getFaceDetectionQRCodeSucLiveData) {bean->
            getJGPushReceiverLiveData.value=bean.merchantLogo
            getJGPushReceiverLiveData//lambda的最后一行要返回一个 LiveData对象
        }



resultLiveData.observe(owner=reference?.get()!!, onChanged = {
            LogUtil.e(TAG, "输出 resultLiveData 数据=${it}") //输出 resultLiveData 数据=https://test-consumer.aiyzy.com//file/image/1580808861459533825
        })

抛Cannot add the same observer with different lifecycles的问题

如果一个activity,在onCreate的时候建立Livedata监听,当此activity启动两遍的时候,会抛出Cannot add the same observer with different lifecycles异常,原因是使用了lamda表达式。

使用lamda表达式建立监听,当Activity未finish的时候,再一次启动的时候,就会报异常,

原因是在同一个类里 使用lamda表达式 实例化的对象,引用外部的静态方法或者静态引用,导致是同一个实例。


public class MyActivity extends AppCompatActivity {
	static String TEST_NAME = "HAHHA";
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
     final int num = 1;
      liveData.observe(this, value-> {
      		// 内部新建实例
           Gson gson = new Gson;
           // 只是使用了外部的基本类型
           int thisNum = num;
           // 调用外部的静态方法和静态引用
           show(TEST_NAME );
         
      });
    }
  static void show(String name){
	.....
  } 
}
   

如何避免这个异常呢?

引用了外部的非静态引用或者非静态的方法,两个相同的activity下,就不是同一个实例,就不会报这个异常错误了。

String EXTAR_NAME = "HAHHA";
public class MyActivity extends AppCompatActivity {

    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
     final int num = 1;
      liveData.observe(this, value-> {
           Gson gson = new Gson;
             // 调用外部静态引用
           String str = EXTAR_NAME;
           // 调用外部非静态方法
           show(EXTAR_NAME);
         
      });
    }
}
 void show(String name){
	.....
}    

  • 8
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
LiveData 是用于在 Android 应用程序中观察数据的一个组件。LiveData 是一个可观察的数据持有者类,它具有生命周期感知能力,因此它会自动管理它与 Activity 和 Fragment 生命周期的关系,从而可以避免内存泄漏和崩溃。 LiveData 的一个重要特性是它可以通知观察者数据已更改。当 LiveData 的值发生更改时,它会自动通知所有观察者。这意味着您无需手动更新 UI 或执行其他操作以反映数据更改。LiveData 还支持数据转换和过滤,因此您可以将原始数据转换为 UI 可以直接使用的格式。 LiveData 使用观察者模式进行数据监听,您可以使用 `observe()` 方法将观察者添加到 LiveData 实例中,该方法需要传入一个 LifecycleOwner 和一个 Observer 对象。LifecycleOwner 表示 LiveData 与哪个组件的生命周期绑定,通常是 Activity 或 Fragment。Observer 对象定义了当 LiveData 的值更改时要执行的操作。 下面是一个简单的示例,演示如何使用 LiveData 监听数据更改: ``` // 创建一个 LiveData 实例 val myLiveData = MutableLiveData<String>() // 将观察者添加到 LiveData 实例中 myLiveData.observe(this, Observer { newValue -> // 在这里更新 UI 或执行其他操作 textView.text = newValue }) // 更改 LiveData 的值 myLiveData.value = "Hello World" ``` 在上面的示例中,我们创建了一个名为 `myLiveData` 的 LiveData 实例,并将其与当前组件的生命周期绑定。我们还将一个 Observer 对象传递给 `observe()` 方法,以便在 LiveData 的值更改时执行操作。当我们调用 `myLiveData.value = "Hello World"` 时,LiveData 会自动通知所有观察者,以便更新 UI 或执行其他操作。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值