在Activity 可见的时候,我们去做一个计数功能,每隔一秒 将计数加1 ,当Activity不可见的时候停止计数,当Activity被销毁的时候 将计数置为0。这里我们新增需求将计数的数字显示在TextView中。
所以我们就要做到当计数的数字发生改变时,通知TextView便于TextView重新显示,如果矬一点,可能会想到将View传递到ViewModel中,让ViewModel持有View的引用,这种方式确实可以实现需求,但是后患无穷,并且View和ViewModel之前只能是单项的,即只能View层持有ViewModel,那么如何优雅的实现找那个需求呢?这就是我们今天说的LiveData了
我们在activity_main3中新增一个TextView用来显示计数
<TextView
android:gravity=“center”
android:layout_margin=“10dp”
android:textSize=“20dp”
android:id=“@+id/tv_count”
android:layout_width=“match_parent”
android:layout_height=“wrap_content”/>
LiveData需要结合ViewModel来使用,之前的Main3ActivityModel代码如下所示:
public class Main3ActivityViewModel extends ViewModel {
public int count = 0;
public Main3ActivityViewModel(int count) {
this.count = count;
}
}
我们将count类型的变量修改为LiveData的类型 代码如下所示:
public class Main3ActivityViewModel extends ViewModel {
public MutableLiveData mCount = new MutableLiveData<>();
public Main3ActivityViewModel(int count) {
this.mCount.setValue(count);
}
}
LiveData类型的变量我们通过set和get去赋值和取值
现在计数的数字已经是LiveData类型的了,那么我们如何在数据变化的时候通知textView呢
我们只需要在Main3Activity中进行如下注册:
main3ActivityViewModel.count.observeForever(new Observer() {
@Override
public void onChanged(Integer integer) {
}
});
光是这样还是不行的,我们还需要修改WorkUtil中的逻辑,因为现在是可变类型的数据,所以我们要将值的改变放在ViewModel中
public class Main3ActivityViewModel extends ViewModel {
public MutableLiveData mCount = new MutableLiveData<>();
public void add() {
mCount.setValue(mCount.getValue() + 1);
}
public Main3ActivityViewModel(int count) {
this.mCount.setValue(count);
}
}
这样的话 我们在WorkUtil中调用add方法 就可以改变mCount的值了,最后我们在注册回到的onChanged方法中去给Textview赋值就可以了,我们运行程序结果如下:
嚯嚯,pia pia 打脸
这里报错的原因是因为我们的计数demo是运行在子线程中的,而LiveData的setValue方法只能在主线程中调用,如果想要在子线程中调动只能使用postValue方法,我们将赋值方法改为postValue,再次运行结果如下所示:
我不知道mac下是否有类似screenToGif这种软件,所以只能截静态图了。
ok,这样的话 我们就使用LiveData实现上面的需求了,但是有没有感觉有什么问题呢,问题就是这个mCount可变类型的数据暴露给了外部,导致我们在ViewModel外也是可以赋值的,这样违反了ViewModel数据的封装性,所以我们需要将这个可变类型的变量声明为私有的并且声明一个不可变的变量赋值给mCount,只对外暴露不可变的LiveData,修改model代码如下所示:
private LiveData count;
public LiveData getCount() {
return mCount;
}
public void setCount(LiveData count) {
this.count = count;
}
private MutableLiveData mCount = new MutableLiveData<>();
public void add() {
mCount.postValue(mCount.getValue() + 1);
}
public Main3ActivityViewModel(int count) {
this.mCount.setValue(count);
}
修改WorkUtil中的访问方法如下所示:
@OnLifecycleEvent(Lifecycle.Event.ON_RESUME)
public void start() {
new Thread(new Runnable() {
@Override
public void run() {
while (whetherToCount) {
try {
Thread.sleep(1000);
main3ActivityViewModel.add();
Log.d(TAG, "start: " + main3ActivityViewModel.getCount());
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}).start();
}
这样一来 就是LiveData的规范用法了。
转换LiveData
.map转换
为了说明转化的作用,我们新建一个Student类,类中有如下字段 :
public class Student {
/**
- 学号
*/
private int stuNumber;
/**
- 姓名
*/
private int stuName;
/**
- 分数
*/
private int stuScore;
…
}
我们新建Main4Activity 对应页面输入分数、保存、显示分数
需求如下:
在输入框中输入分数、在textview中显示分数
<EditText
android:id=“@+id/ed_socre”
android:layout_width=“match_parent”
android:layout_height=“wrap_content”
android:hint=“enter stunumber” />
<Button
android:id=“@+id/btn_save”
android:layout_width=“match_parent”
android:layout_height=“wrap_content”
android:text=“save” />
<TextView
android:gravity=“center”
android:layout_margin=“10dp”
android:textSize=“20sp”
android:id=“@+id/tv_stuscore”
android:layout_width=“match_parent”
android:layout_height=“wrap_content” />
新建对应的Main4ActivityModel 来观察student数据的改变
public class Main4ActivityViewModel extends ViewModel {
private LiveData liveData;
public LiveData getLiveData() {
return studentMutableLiveData;
}
private MutableLiveData studentMutableLiveData = new MutableLiveData<>();
public void setStudentMutableLiveData(Student studentMutableLiveData) {
this.studentMutableLiveData.setValue(studentMutableLiveData);
}
}
我们这里直接使用setStudentMutableLiveData来模拟数据的获取,正常情况下我们需要在ViewModel去请求网络数据进行设置
我们在Main4Activity中直接进行数据设置操作:
private Main4ActivityViewModel main4ActivityViewModel;
private Button btnSave;
private TextView tvScore;
private EditText edScore;
private Student student = new Student();
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main4);
btnSave = findViewById(R.id.btn_save);
tvScore = findViewById(R.id.tv_stuscore);
edScore = findViewById(R.id.ed_socre);
main4ActivityViewModel = new ViewModelProvider(this).get(Main4ActivityViewModel.class);
student.setStuName(“黄林晴”);
student.setStuNumber(20200522);
最后
感觉现在好多人都在说什么安卓快凉了,工作越来越难找了。又是说什么程序员中年危机啥的,为啥我这年近30的老农根本没有这种感觉,反倒觉得那些贩卖焦虑的都是瞎j8扯谈。当然,职业危机意识确实是要有的,但根本没到那种草木皆兵的地步好吗?
Android凉了都是弱者的借口和说辞。虽然 Android 没有前几年火热了,已经过去了会四大组件就能找到高薪职位的时代了。这只能说明 Android 中级以下的岗位饱和了,现在高级工程师还是比较缺少的,很多高级职位给的薪资真的特别高(钱多也不一定能找到合适的),所以努力让自己成为高级工程师才是最重要的。
所以,最后这里放上我耗时两个月,将自己8年Android开发的知识笔记整理成的Android开发者必知必会系统学习资料笔记,上述知识点在笔记中都有详细的解读,里面还包含了腾讯、字节跳动、阿里、百度2019-2021面试真题解析,并且把每个技术点整理成了视频和PDF(知识脉络 + 诸多细节)。
以上全套学习笔记面试宝典,吃透一半保你可以吊打面试官,只有自己真正强大了,有核心竞争力,你才有拒绝offer的权力,所以,奋斗吧!骚年们!千里之行,始于足下。种下一颗树最好的时间是十年前,其次,就是现在。
最后,赠与大家一句诗,共勉!
不驰于空想,不骛于虚声。不忘初心,方得始终。
网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。
一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
里面还包含了腾讯、字节跳动、阿里、百度2019-2021面试真题解析,并且把每个技术点整理成了视频和PDF(知识脉络 + 诸多细节)。
[外链图片转存中…(img-HHSPvbIb-1714282254917)]
以上全套学习笔记面试宝典,吃透一半保你可以吊打面试官,只有自己真正强大了,有核心竞争力,你才有拒绝offer的权力,所以,奋斗吧!骚年们!千里之行,始于足下。种下一颗树最好的时间是十年前,其次,就是现在。
最后,赠与大家一句诗,共勉!
不驰于空想,不骛于虚声。不忘初心,方得始终。
网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。
一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!