Android-ViewModel和LiveData使用詳解
ViewModel類的設(shè)計(jì)目的是以一種關(guān)注生命周期的方式存儲(chǔ)和管理與UI相關(guān)的數(shù)據(jù)。
例如:Activity在配置發(fā)生改變時(shí)(屏幕旋轉(zhuǎn)),Activity就會(huì)重新創(chuàng)建,onCreate()方法也會(huì)重新調(diào)用。我們可以在onSaveInstanceState()方法中保存數(shù)據(jù),并從onCreate()方法中通過(guò)Bundle恢復(fù)數(shù)據(jù),但這種方法只適用于可以對(duì)其進(jìn)行序列化的少量數(shù)據(jù),而不適用于潛在的大量數(shù)據(jù)。使用ViewModel的話ViewModel會(huì)自動(dòng)保留之前的數(shù)據(jù)并給新的Activity或Fragment使用。直到當(dāng)前Activity被系統(tǒng)銷毀時(shí),F(xiàn)ramework會(huì)調(diào)用ViewModel的onCleared()方法,我們可以在onCleared()方法中做一些資源清理操作。
LiveData是一個(gè)可觀察的數(shù)據(jù)持有者類。與常見(jiàn)的觀察者不同,LiveData是有生命周期感知的。這意味著它尊重其他應(yīng)用程序組件的生命周期,比如Activity、Fragment或Service。這種感知確保LiveData只更新處于生命周期狀態(tài)內(nèi)的應(yīng)用程序組件。
LiveData是由observer類表示的觀察者視為處于活動(dòng)狀態(tài),如果其生命周期處于STARTED或RESUMED狀態(tài)。LiveData會(huì)將觀察者視為活動(dòng)狀態(tài),并通知其數(shù)據(jù)的變化。LiveData未注冊(cè)的觀察對(duì)象以及非活動(dòng)觀察者是不會(huì)收到有關(guān)更新的通知。
LiveData的優(yōu)點(diǎn):
確保UI界面的數(shù)據(jù)狀態(tài)
LiveData遵循觀察者模式。LiveData在生命周期狀態(tài)更改時(shí)通知Observer對(duì)象,更新這些Observer對(duì)象中的UI。觀察者可以在每次應(yīng)用程序數(shù)據(jù)更改時(shí)更新UI,而不是每次發(fā)生更改時(shí)更新UI。
沒(méi)有內(nèi)存泄漏
當(dāng)觀察者被綁定他們對(duì)應(yīng)的LifeCycle以后,當(dāng)頁(yè)面銷毀時(shí)他們會(huì)自動(dòng)被移除,不會(huì)導(dǎo)致內(nèi)存溢出。
不會(huì)因?yàn)锳ctivity的不可見(jiàn)導(dǎo)致Crash
當(dāng)Activity不可見(jiàn)時(shí),即使有數(shù)據(jù)變化,LiveData也不會(huì)通知觀察者。因?yàn)榇藭r(shí)觀察者的LifeCyele并不處于Started或者RESUMED狀態(tài)。
配置的改變
當(dāng)前Activity配置改變(如屏幕方向),導(dǎo)致重新從onCreate走一遍,這時(shí)觀察者們會(huì)立刻收到配置變化前的最新數(shù)據(jù)。
不用再人為的處理生命周期
Activity或者Fragment只要在需要觀察數(shù)據(jù)的時(shí)候觀察數(shù)據(jù)即可,不需要理會(huì)生命周期變化了。這一切都交給LiveData來(lái)自動(dòng)管理。
添加ViewModel和LiveData庫(kù)的依賴
//build.gradle文件中
allprojects {
repositories {
google()
jcenter()
}
}
//app/build.gradle文件中
implementation 'androidx.lifecycle:lifecycle-extensions:2.0.0-alpha1'
定義ViewModel和創(chuàng)建LiveData
package io.dcloud.H56580E2E.viewModelLiveData;
import android.app.Application;
import androidx.annotation.NonNull;
import androidx.lifecycle.AndroidViewModel;
import androidx.lifecycle.MutableLiveData;
import androidx.lifecycle.ViewModel;
/**
* ViewModel 主要解決的是Fragment的參數(shù)傳遞問(wèn)題
* ViewMode和對(duì)象實(shí)例綁定,也就是說(shuō)ViewModel不會(huì)由于ConfigurationChange而改變。比如:在你的手機(jī)進(jìn)行翻轉(zhuǎn)時(shí),activity會(huì)重新走生命周期。
* 那么如果在Activity中保存的數(shù)據(jù),就很有可能,被復(fù)寫,重置或者丟失。但是如果我們將activity當(dāng)作View組件的話,那么我們的關(guān)鍵屬性就會(huì)自然而然的存放在ViewModel中。
* 這時(shí)候如果acitivty的ConfigrationChange調(diào)用的話,由于activity對(duì)象并沒(méi)有被重建,還是之前的對(duì)象,那么我們所取到的ViewModel也不會(huì)發(fā)生變化
*/
/**
* 定義ViewModel和創(chuàng)建LiveData
* ViewModel是以關(guān)聯(lián)生命周期的方式來(lái)存儲(chǔ)和管理UI相關(guān)的數(shù)據(jù)的類,即使configuration發(fā)生改變(比如旋轉(zhuǎn)屏幕),數(shù)據(jù)仍然可以存在不會(huì)銷毀(使用ViewModel就繼承 AndroidViewModel/ViewModel )
* AndroidViewModel(ViewModel的子類):需要使用到Context對(duì)象時(shí)繼承它 否則繼承 ViewModel
*
* LiveData是一個(gè)可觀察的數(shù)據(jù)持有者類
* ViewModel 與 LiveData相互配合使用
*/
public class DomeModel extends ViewModel {
//創(chuàng)建LiveData(可以創(chuàng)建多個(gè)不同類型的 LiveData..)
private MutableLiveData<DomeInfo> mDomeLiveData;
/**
* 改變 LiveData 中的數(shù)據(jù)
* 使用 setValue()
* @param phone_str
* @param pwd_str
* setValue()要在主線程中調(diào)用
*/
public void setDomeInfo(String phone_str,String pwd_str){
mDomeLiveData.setValue(new DomeInfo(phone_str,pwd_str));
}
/**
* 改變 LiveData 中的數(shù)據(jù)
* 使用 postValue()
* postValue()既可在主線程也可在子線程中調(diào)用
*/
public void postDomeInfo(String phone_str,String pwd_str){
mDomeLiveData.postValue(new DomeInfo(phone_str,pwd_str));
}
/**
*
* @return
*/
public MutableLiveData<DomeInfo> getmDomeLiveData(){
if(mDomeLiveData == null){
mDomeLiveData = new MutableLiveData<>();
}
return mDomeLiveData;
}
// 當(dāng)MyActivity被銷毀時(shí),F(xiàn)ramework會(huì)調(diào)用ViewModel的onCleared()
@Override
protected void onCleared() {
super.onCleared();
}
}
使用(演示在 Activity中與Fragment中和Fragment與Fragment相互通信)
在Activity中:
package io.dcloud.H56580E2E.viewModelLiveData;
import androidx.appcompat.app.AppCompatActivity;
import androidx.lifecycle.Observer;
import androidx.lifecycle.ViewModelProviders;
import butterknife.BindView;
import butterknife.ButterKnife;
import io.dcloud.H56580E2E.R;
import io.reactivex.functions.Consumer;
import android.os.Bundle;
import android.widget.Button;
import android.widget.TextView;
import com.jakewharton.rxbinding3.view.RxView;
import java.util.concurrent.TimeUnit;
public class DomeActivity extends AppCompatActivity {
@BindView(R.id.button)
Button addBut;
@BindView(R.id.textView3)
TextView showData_text;
//ViewModel 對(duì)象
private DomeModel domeModel;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_dome);
ButterKnife.bind(this);
//添加 Fragment視圖
getSupportFragmentManager().beginTransaction().replace(R.id.frameLayout1,new OneFragment()).commit();
getSupportFragmentManager().beginTransaction().replace(R.id.frameLayout2,new TwoFragment()).commit();
//初始化 ViewModel
domeModel = ViewModelProviders.of(this).get(DomeModel.class);
//Button按鈕點(diǎn)擊事件
RxView.clicks(addBut).throttleFirst(2, TimeUnit.SECONDS)
.subscribe(new Consumer<Object>() {
@Override
public void accept(Object unit) throws Exception {
//使用 postValue()方法更新一條數(shù)據(jù)
domeModel.postDomeInfo("13233253173","11111111");
}
});
//獲取 上面更新的數(shù)據(jù)
domeModel.getmDomeLiveData().observe(this, new Observer<DomeInfo>() {
@Override
public void onChanged(DomeInfo domeInfo) {
showData_text.setText("賬號(hào):"+domeInfo.getPhone_str()+"密碼:"+domeInfo.getPwd_str());
}
});
}
}
Fragment 1:
package io.dcloud.H56580E2E.viewModelLiveData;
import androidx.lifecycle.Observer;
import androidx.lifecycle.ViewModelProviders;
import android.os.Bundle;
import androidx.annotation.NonNull;
import androidx.annotation.Nullable;
import androidx.fragment.app.Fragment;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.Button;
import android.widget.TextView;
import com.jakewharton.rxbinding3.view.RxView;
import java.util.concurrent.TimeUnit;
import butterknife.BindView;
import butterknife.ButterKnife;
import butterknife.Unbinder;
import io.dcloud.H56580E2E.R;
import io.reactivex.functions.Consumer;
public class OneFragment extends Fragment {
//ButterKnife對(duì)象
private Unbinder unbinder;
//創(chuàng)建 ViewModel 對(duì)象
private DomeModel domeModel;
@BindView(R.id.textView)
TextView show_textview;
@BindView(R.id.button2)
Button update_but;
public OneFragment() {
}
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
}
@Override
public View onCreateView(@NonNull LayoutInflater inflater, @Nullable ViewGroup container,
@Nullable Bundle savedInstanceState) {
View view= inflater.inflate(R.layout.one_fragment, container, false);
unbinder=ButterKnife.bind(this,view);
return view;
}
/*onViewCreated是在onCreateView后被觸發(fā)的事件*/
@Override
public void onViewCreated(@NonNull View view, @Nullable Bundle savedInstanceState) {
super.onViewCreated(view, savedInstanceState);
}
@Override
public void onActivityCreated(@Nullable Bundle savedInstanceState) {
super.onActivityCreated(savedInstanceState);
domeModel=ViewModelProviders.of(getActivity()).get(DomeModel.class);
//在 oneFragment中更新數(shù)據(jù)
RxView.clicks(update_but).throttleFirst(2,TimeUnit.SECONDS)
.subscribe(new Consumer<Object>() {
@Override
public void accept(Object unit) throws Exception {
domeModel.postDomeInfo("fragement1323325317","123456");
}
});
// 獲取更新的數(shù)據(jù)(結(jié)合 RxJava2 操作符)
domeModel.getmDomeLiveData().observe(this, new Observer<DomeInfo>() {
@Override
public void onChanged(DomeInfo domeInfo) {
show_textview.setText("賬號(hào):"+domeInfo.getPhone_str()+"密碼:"+domeInfo.getPwd_str());
}
});
}
@Override
public void onDestroyView() {
super.onDestroyView();
//Fragment銷毀的時(shí)候解綁 ButterKnife
unbinder.unbind();
}
}
Fragment 2:
package io.dcloud.H56580E2E.viewModelLiveData;
import android.content.Context;
import android.os.Bundle;
import androidx.annotation.Nullable;
import androidx.fragment.app.Fragment;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.Button;
import android.widget.TextView;
import com.jakewharton.rxbinding3.view.RxView;
import java.util.concurrent.TimeUnit;
import androidx.lifecycle.Observer;
import androidx.lifecycle.ViewModelProviders;
import butterknife.BindView;
import butterknife.ButterKnife;
import butterknife.Unbinder;
import io.dcloud.H56580E2E.R;
import io.reactivex.functions.Consumer;
/**
* A simple {@link Fragment} subclass.
* Activities that contain this fragment must implement the
* create an instance of this fragment.
*/
public class TwoFragment extends Fragment {
@BindView(R.id.textView5)
TextView show_textview;
@BindView(R.id.button3)
Button button;
private DomeModel domeModel;
private Unbinder unbinder;
public TwoFragment() {
// Required empty public constructor
}
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
}
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
Bundle savedInstanceState) {
// Inflate the layout for this fragment
View view= inflater.inflate(R.layout.fragment_two, container, false);
unbinder=ButterKnife.bind(this,view);
return view;
}
@Override
public void onActivityCreated(@Nullable Bundle savedInstanceState) {
super.onActivityCreated(savedInstanceState);
domeModel = ViewModelProviders.of(getActivity()).get(DomeModel.class);
RxView.clicks(button).throttleFirst(2,TimeUnit.SECONDS)
.subscribe(new Consumer<Object>() {
@Override
public void accept(Object unit) throws Exception {
domeModel.postDomeInfo("fragement2-1323325317","123456");
}
});
// 獲取更新的數(shù)據(jù)(結(jié)合 RxJava2 操作符)
domeModel.getmDomeLiveData().observe(this, new Observer<DomeInfo>() {
@Override
public void onChanged(DomeInfo domeInfo) {
show_textview.setText("賬號(hào):"+domeInfo.getPhone_str()+"密碼:"+domeInfo.getPwd_str());
}
});
}
@Override
public void onAttach(Context context) {
super.onAttach(context);
}
@Override
public void onDetach() {
super.onDetach();
}
@Override
public void onDestroyView() {
super.onDestroyView();
//Fragment銷毀的時(shí)候解綁 ButterKnife
unbinder.unbind();
}
}
Dome實(shí)體類:
package io.dcloud.H56580E2E.viewModelLiveData;
/**
* Dome實(shí)體類
*/
public class DomeInfo {
private String phone_str;
private String pwd_str;
public DomeInfo(String phone_str, String pwd_str) {
this.phone_str = phone_str;
this.pwd_str = pwd_str;
}
public DomeInfo() {
}
public String getPhone_str() {
return phone_str;
}
public void setPhone_str(String phone_str) {
this.phone_str = phone_str;
}
public String getPwd_str() {
return pwd_str;
}
public void setPwd_str(String pwd_str) {
this.pwd_str = pwd_str;
}
}
以上這篇Android-ViewModel和LiveData使用詳解就是小編分享給大家的全部?jī)?nèi)容了,希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
- Android Jetpack架構(gòu)組件Lifecycle詳解
- Android Jetpack架構(gòu)組件 ViewModel詳解
- Android 生命周期架構(gòu)組件使用方法
- Android架構(gòu)組件Room的使用詳解
- Android架構(gòu)組件Room指南
- Android LiveData使用需要注意的地方
- Android mvvm之LiveData原理案例詳解
- Android 基于MediatorLiveData實(shí)現(xiàn)紅點(diǎn)的統(tǒng)一管理
- 詳解Android JetPack之LiveData的工作原理
- Android架構(gòu)組件LiveData使用詳解
相關(guān)文章
Android 使用Vitamio打造自己的萬(wàn)能播放器(5)——在線播放(播放優(yōu)酷視頻)
本文主要介紹Android Vitamio的使用,這里給大家提供效果圖和代碼實(shí)例,來(lái)說(shuō)明Vitamio組件播放網(wǎng)絡(luò)視頻,有需要的小伙伴可以參考下2016-07-07
自定義toast外形,多次點(diǎn)擊不會(huì)總是彈出toast的實(shí)現(xiàn)方法
下面小編就為大家?guī)?lái)一篇自定義toast外形,多次點(diǎn)擊不會(huì)總是彈出toast的實(shí)現(xiàn)方法。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-04-04
Android中Item實(shí)現(xiàn)點(diǎn)擊水波紋效果
這篇文章主要給大家介紹了關(guān)于Android中Item實(shí)現(xiàn)點(diǎn)擊水波紋效果的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)各位Android開(kāi)發(fā)者們具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2018-11-11
android 獲取上一個(gè)activity返回值的方法
android 獲取上一個(gè)activity返回值的方法,需要的朋友可以參考一下2013-06-06
Android如何添加控件監(jiān)聽(tīng)器(三種方式)
本文主要介紹了Android如何添加控件監(jiān)聽(tīng)器(三種方式),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2022-06-06
Android程序開(kāi)發(fā)之Listview下拉刷新上拉(滑動(dòng)分頁(yè))加載更多
這篇文章主要介紹了Android程序開(kāi)發(fā)之Listview下拉刷新上拉(滑動(dòng)分頁(yè))加載更多的相關(guān)資料,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下2016-07-07
flutter 微信聊天輸入框功能實(shí)現(xiàn)
這篇文章主要介紹了flutter 微信聊天輸入框功能實(shí)現(xiàn),本文通過(guò)實(shí)例圖文相結(jié)合給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-03-03

