JetPack之LiveData

目录

  • 一、LiveData简介
    • 1.1 LiveData是什么?
  • 二、LiveData使用
    • 2.1 LiveData基础使用
    • 2.2 LiveData搭配Service模拟后台消息
    • 2.3 LiveData在组件中的数据传递
  • 三、LiveData应用场景


一、LiveData简介

1.1 LiveData是什么?

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

LiveData逻辑流程
在这里插入图片描述

LiveData作用:

实时刷新数据
防止内存泄漏

LiveData 采用的是观察者模式,当 LiveData 保存的数据发生变化时就会通知观察者,观察者接收到通知后可以进行 UI 数据刷新或者其他操作。
那它是怎么做到防止内存泄漏的呢 ?在给 LiveData 添加观察者对象的时候可以绑定一个具有生命周期的组件,当组件生命周期处于活跃状态(即 STARTED 、RESUMED 状态)时数据更新才会通知观察者,当组件被销毁时则会自动移除对应的观察者对象,从而防止一直持有对应组件防止内存泄漏。


二、LiveData使用

在 Android 中,LiveData 提供了两种方法来更新数据:postValue 和 setValue。

postValue
postValue 方法用于在非主线程中更新 LiveData 的值。
在后台线程中使用 postValue 方法更新 LiveData 的值时,LiveData 会确保数据更新操作在主线程中执行,以避免在非主线程中直接更新 UI 导致的问题。
postValue 方法是线程安全的,可以在任何线程中调用。
setValue
setValue 方法用于在主线程中更新 LiveData 的值。
在主线程以外的线程(如后台线程)中调用 setValue 方法,会导致 IllegalStateException,因为直接在非主线程中更新 LiveData 的值可能导致 UI 不同步等问题。setValue 方法应该只在主线程中调用。

postValue 和 setValue 的主要区别在于线程安全性和线程限制:

postValue 可以在任何线程中调用,内部会确保在主线程中更新数据,适合在后台线程中更新数据。
setValue 只能在主线程中调用,用于在主线程中更新数据。

2.1 LiveData基础使用

搭配单例使用getInfo1()初始化LiveData
myLiveData

public class myLiveData {
    private static MutableLiveData<String> info1;

    public static MutableLiveData<String> getInfo1() {
        if (info1 == null) {
            info1 = new MutableLiveData<>();
        }
        return info1;
    }
}

TextView根据myLiveData发送来的消息进行UI更新。
LiveDataActivity

public class LiveDataActivity extends AppCompatActivity {

    TextView data;

    @SuppressLint("MissingInflatedId")
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_live_data2);
        data = findViewById(R.id.LiveData2);


        //1.观察者 眼睛 环节
        myLiveData.getInfo1().observe(this, new Observer<String>() {
            @Override
            public void onChanged(String s) {
                data.setText(s); // 更新 UI
            }
        });

        //2. 触发数据改变环节
        myLiveData.getInfo1().setValue("--------default------------");//主线程修改数据
        new Thread(new Runnable() {
            @Override
            public void run() {
                try {
                    Thread.sleep(3000);
                    myLiveData.getInfo1().postValue("--------3秒后修改UI------------");
                    Thread.sleep(6000);
                    myLiveData.getInfo1().postValue("--------6秒后修改UI------------");
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        }).start();
    }
}

看下效果
在这里插入图片描述

2.2 LiveData搭配Service模拟后台消息

按钮开启服务,当界面可见后使用Toast模拟更新UI
LiveDataActivity

public class LiveDataActivity extends AppCompatActivity {
    String TAG = "Henry";
    TextView data;
    Button button;

    @SuppressLint("MissingInflatedId")
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_live_data2);
        data = findViewById(R.id.LiveData2);
        button = findViewById(R.id.livedata_Button);

        test2();
    }
    public void test2() {
        button.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                startService(new Intent(LiveDataActivity.this, MyService.class));
                Toast.makeText(LiveDataActivity.this, "启动服务成功", Toast.LENGTH_SHORT).show();
            }
        });
        //观察者,界面可见的情况下才会下列事情
        myLiveData.getInfo1().observe(this, new Observer<String>() {
            @Override
            public void onChanged(String s) {
                Log.d(TAG, "界面可见,当前用户正在查看微信列表界面,更新消息列表UI界面" + s);
                Toast.makeText(LiveDataActivity.this, "更新UI界面成功" + s, Toast.LENGTH_SHORT).show();
            }
        });

    }
  }

当服务开启使用LiveData后台发送消息
MyService

public class MyService extends Service {
    String TAG="Henry";
    public MyService() {
    }

    @Override
    public IBinder onBind(Intent intent) {
        // TODO: Return the communication channel to the service.
        throw new UnsupportedOperationException("Not yet implemented");
    }

    @Override
    public int onStartCommand(Intent intent, int flags, int startId) {
        new Thread(new Runnable() {
            @Override
            public void run() {
                for(int i=0;i<10000;i++){
                    Log.d(TAG,"消息铃声--------------");
                    myLiveData.getInfo1().postValue("---------消息内容"+i);
                    try {
                        Thread.sleep(5000);
                    } catch (InterruptedException e) {
                        throw new RuntimeException(e);
                    }
                }
            }
        }).start();
        return super.onStartCommand(intent, flags, startId);
    }
}

测试一下:
在这里插入图片描述

2.3 LiveData在组件中的数据传递

在Activity的跳转后,LiveData是否还能获取之前的消息?
LiveDataActivity

public class LiveDataActivity extends AppCompatActivity {
    String TAG = "Henry";
    TextView data;
    Button button;
    Button jump;

    @SuppressLint("MissingInflatedId")
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_live_data2);
        data = findViewById(R.id.LiveData2);
        button = findViewById(R.id.livedata_Button);
        jump= findViewById(R.id.livedata_jump);
        test3();
    }


    public void test3() {
        jump.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
            //旧数据
                myLiveData.getInfo1().setValue("this message is from LiveDataActivity");
                startActivity(new Intent(LiveDataActivity.this,
                        LiveDataSecondActivity.class));
            }
        });
  }

LiveDataSecondActivity

public class LiveDataSecondActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_live_data_second);

        //后观察数据 可以收到前面的数据
        myLiveData.getInfo1().observe(this, new Observer<String>() {
            @Override
            public void onChanged(String s) {
                Toast.makeText(LiveDataSecondActivity.this, "观察数据变化 消息为=    " + s,
                        Toast.LENGTH_SHORT).show();
            }
        });
        //现在的新数据
        myLiveData.getInfo1().postValue("new Value");
    }
}

测试一下:LiveData可以在组件中传递,先setvalue,后订阅,也可以收到数据,这就是粘性数据
在这里插入图片描述


三、LiveData应用场景

LiveData 是一个具有生命周期感知能力的数据持有类,通常与 ViewModel 结合使用,用于在应用程序中管理和展示数据。常见应用场景如下:

1.UI 更新:LiveData 可以用于在数据发生变化时更新 UI。通过观察 LiveData 对象,可以实时更新界面上的数据,而且 LiveData 会确保数据更新是在主线程中进行的,避免了线程安全问题。
2.数据缓存:LiveData 可以用于缓存数据,避免每次 UI 重建时都需要重新加载数据。当数据发生变化时,LiveData 会通知观察者,从而实现数据的实时更新。
3.网络请求:在进行网络请求时,可以使用 LiveData 来持有请求结果。当数据加载完成后,更新 LiveData 的值,界面上的数据会自动更新。
4.数据共享:LiveData 可以在多个组件之间共享数据,而且只需一个数据源。这样可以确保数据的一致性,并且避免了数据同步的问题。
5.状态管理:LiveData 可以用于管理应用程序的状态,例如加载中、成功、错误等状态。通过更新 LiveData 的值,可以通知 UI 层当前应用程序的状态。
6.与 Room 结合使用:LiveData 可以与 Room 数据库一起使用,实现数据持久化和实时更新。Room 数据库中的查询结果可以返回 LiveData 对象,从而实现数据的观察和自动更新。

参考链接:
关于LiveData粘性事件所带来问题的解决方案
Android Jetpack 之 LiveData 详解
LiveData 粘性事件(原理+四个解决方法)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/474621.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

这里是一本关于 DevOps 企业级 CI/CD 实战的书籍...

文章目录 &#x1f4cb; 前言&#x1f3af; 什么是 DevOps&#x1f3af; 什么是 CI/CD&#x1f3af;什么是 Jenkins&#x1f9e9; Jenkins 简单案例 &#x1f3af; DevOps 企业级实战书籍推荐&#x1f525; 参与方式 &#x1f4cb; 前言 企业级 CI/CD 实战是一个涉及到软件开发…

语音神经科学—05. Human cortical encoding of pitch in tonal and non-tonal languages

Human cortical encoding of pitch in tonal and non-tonal languages&#xff08;在音调语音和非音调语言中人类大脑皮层的音高编码&#xff09; 专业术语 tonal language 音调语言 pitch 音高 lexical tone 词汇音调 anatomical properties 解刨学特性 temporal lobe 颞叶 s…

开源Thinkphp核心在线网页音乐播放php源码,附带系统搭建教程

安装教程 环境要求&#xff1a;apachePHP7.0Thinkphp伪静态 安装教程&#xff1a;修改Application目录下的database.php信息 导入根目录下的install.sql到数据库 修改Static目录下的player目录下的player.js文件的第140行的“域名”为你的域名 修改Static目录下的player2目录下…

四川易点慧电子商务抖音小店:安全可靠,购物新选择

在数字化浪潮席卷全球的今天&#xff0c;电子商务已成为人们日常生活中不可或缺的一部分。四川易点慧电子商务抖音小店&#xff0c;作为新兴的电商力量&#xff0c;以其安全可靠的特点&#xff0c;赢得了广大消费者的信赖和喜爱。 一、品牌信誉&#xff0c;品质保障 四川易点慧…

【MySQL】数据库的基础概念

&#x1f466;个人主页&#xff1a;Weraphael ✍&#x1f3fb;作者简介&#xff1a;目前学习计网、mysql和算法 ✈️专栏&#xff1a;MySQL学习 &#x1f40b; 希望大家多多支持&#xff0c;咱一起进步&#xff01;&#x1f601; 如果文章对你有帮助的话 欢迎 评论&#x1f4ac…

Jackson 2.x 系列【4】对象映射器 ObjectMapper

有道无术&#xff0c;术尚可求&#xff0c;有术无道&#xff0c;止于术。 本系列Jackson 版本 2.17.0 源码地址&#xff1a;https://gitee.com/pearl-organization/study-jackson-demo 文章目录 1. 概述2. 案例演示2.1 创建对象2.2 写入2.3 读取 3. 泛型擦除 1. 概述 在前两篇…

【漏洞复现】5. Fastjson 1.2.24反序列化漏洞(CVE-2017-18349)复现

文章目录 1. 预备知识2. 漏洞复现2.1 漏洞介绍2.2 漏洞原理分析2.2.1 Fastjson序列化/反序列化原理2.2.2 Fastjson反序列化漏洞原理 2.3 实验环境2.3.1 靶场搭建 2.3.2 攻击机配置2.3.3 Java反序列化工具marshalsec&#xff1a;2.4 漏洞复现2.4.1 漏洞探测 2.5 漏洞修复 1. 预备…

中央空调的计费方式

中央空调如何计费 电费计量型中央空调计费方法 计费原理:电费计量型就是通过计量空调末端的用电量&#xff0c;再根据用电量换算为冷量&#xff0c;统计中央空调系统中各用户的总冷量&#xff0c;再根据各用户的冷量比例来分摊费用。 优点: 电量参数容易计量&#xff0c;管理…

白话transformer(五):位置编码

在前面其实讲过位置编码的完整内容&#xff0c;这次我们具体看看他的数学原理 B站视频讲解 白话transformer&#xff08;五&#xff09; 1、位置编码的位置 根据原论文的结构图我们可以看到&#xff0c;位置编码位于embedding后&#xff0c;在正式进入注意力机制前面。 也就是…

【二叉树】算法例题

目录 九、二叉树 68. 二叉树的最大深度 ① 69. 相同的树 ① √ 70. 翻转二叉树 ① 71. 对称二叉树 ① 72. 从前序与中序遍历序列构造二叉树 ② 73. 从中序与后续遍历序列构造二叉树 ② 74. 填充每个节点的下一个右侧节点指针 II ② 75. 二叉树展开为链表 ② 76.…

蓝牙耳机品牌排行榜前十名,选购硬核机型,避免不必要的开销!

在现代社会&#xff0c;蓝牙耳机正逐渐取代传统有线耳机&#xff0c;成为主流的选择。尽管市场竞争激烈&#xff0c;但找到一款适合自己的蓝牙耳机并非易事。我在这里为你推荐几款我认为值得信赖的蓝牙耳机&#xff0c;希望能助你一臂之力&#xff0c;找到最适合你的那一款。 一…

一文读懂MES和ERP的区别

MES&#xff08;Manufacturing Execution System&#xff09;系统是制造执行系统&#xff0c;位于上层的计划管理系统与生产过程的直接工业控制系统之间&#xff0c;是面向车间层的管理信息系统&#xff0c;能够对整个车间制造过程进行优化&#xff0c;实时收集生产过程中的数据…

天翼云研发告诉我:AH封装的IPsec不能穿越NAT设备

正文共&#xff1a;1333 字 14 图&#xff0c;预估阅读时间&#xff1a;2 分钟 最近跟中国电信做VPN的研发交流了一下技术&#xff0c;发现技术爱好者跟研发之间的差距还是很明显的。 问题是我在配置天翼云的IPsec VPN连接时&#xff0c;发现IPsec策略的传输协议只有ESP协议可选…

YOLOv9改进策略:ECVBlock即插即用的多尺度融合模块,助力小目标涨点 | 顶刊TIP 2023 CFPNet

&#x1f4a1;&#x1f4a1;&#x1f4a1;本文改进内容&#xff1a;ECVBlock即插即用的多尺度融合模块&#xff0c;助力检测任务有效涨点&#xff01; yolov9-c-EVCBlock summary: 1011 layers, 68102630 parameters, 68102598 gradients, 252.4 GFLOPs 改进结构图如下&#x…

Centos7没有可用软件包 ifconfig问题解决

问题描述 在Centos7中查看ip没有ifconfig&#xff0c;使用yum安装ifconfig报错没有可用软件包 ifconfig问题解决 [rootlocalhost etc]# yum -y install ifconfig 已加载插件&#xff1a;fastestmirror base …

外贸网站常用的wordpress模板

零件配件WordPress外贸建站模板 汽车行业零配件WordPress外贸建站模板&#xff0c;卖配件、零件的外贸公司可以使用的WordPress主题。 https://www.jianzhanpress.com/?p4912 WordPress外贸独立站主题 简洁实用的WordPress外贸独立站主题&#xff0c;适合时尚服装行业搭建w…

酷开系统用电视为居家生活打开精彩窗口|酷开科技|酷开会员|

随着互联网的发展&#xff0c;电视也承载了更多的功能。相比于传统的电视&#xff0c;如今的智能电视屏幕更大、分辨率更高、色彩更加鲜艳&#xff0c;能够呈现出更加逼真的画面效果。当观众观看大屏电视时&#xff0c;仿佛置身于电影大幕的场景之中&#xff0c;感受到更为震撼…

SpringCloud Alibaba Nacos 服务注册和配置中心

一、前言 接下来是开展一系列的 SpringCloud 的学习之旅&#xff0c;从传统的模块之间调用&#xff0c;一步步的升级为 SpringCloud 模块之间的调用&#xff0c;此篇文章为第十二篇&#xff0c;即介绍 SpringCloud Alibaba Nacos 服务注册和配置中心。 二、Nacos 简介 2.1 为…

C++初阶:vector相关练习

目录 1. 只出现一次的数2. 杨辉三角3. 删除有序数组中的重复项4. 只出现一次的数II5. 只出现一次的数III6. 数组中出现次数超过一半的数7. 电话号码的字母组合&#xff08;多叉树遍历&#xff09; 1. 只出现一次的数 题目信息&#xff1a; 题目链接&#xff1a; 只出现一次的数…

【python】flask服务端响应与重定向处理

✨✨ 欢迎大家来到景天科技苑✨✨ &#x1f388;&#x1f388; 养成好习惯&#xff0c;先赞后看哦~&#x1f388;&#x1f388; &#x1f3c6; 作者简介&#xff1a;景天科技苑 &#x1f3c6;《头衔》&#xff1a;大厂架构师&#xff0c;华为云开发者社区专家博主&#xff0c;…