JetPack之LiveData

news2024/11/18 8:34:14

目录

  • 一、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 粘性事件(原理+四个解决方法)

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

[数据结构初阶]二叉树

各位读者老爷好&#xff0c;鼠鼠我现在浅浅介绍一些关于二叉树的知识点&#xff0c;在各位老爷茶余饭后的闲暇时光不妨看看&#xff0c;鼠鼠很希望得到各位老爷的指正捏&#xff01; 开始介绍之前&#xff0c;给各位老爷看一张风景照&#xff0c;有读者老爷知道在哪里吗&#x…

02-MySQL数据库的基本使用与密码设置

一、服务端口 3306端口和33060端口&#xff0c;是我们启动数据库后开启的监听端口&#xff1b; 3306端口&#xff1a;是我们MySQL服务的监听端口&#xff0c;用来连接数据库使用&#xff1b; 33060端口&#xff1a;MySQL-shell服务的端口&#xff0c;MySQL-shell是MySQL架构集群…

基于霍夫检测(hough变换)的人眼瞳孔定位,Matlab实现

博主简介&#xff1a; 专注、专一于Matlab图像处理学习、交流&#xff0c;matlab图像代码代做/项目合作可以联系&#xff08;QQ:3249726188&#xff09; 个人主页&#xff1a;Matlab_ImagePro-CSDN博客 原则&#xff1a;代码均由本人编写完成&#xff0c;非中介&#xff0c;提供…

Java设计模式 | 抽象工厂模式

抽象工厂模式 工厂方法模式中考虑的是一类产品的生产&#xff0c;如幼儿园只培养小朋友&#xff0c;鞋厂只生产鞋子。这些工厂只生产同种类产品&#xff0c;同种类产品称为同等级产品&#xff0c;即工厂方法模式只考虑生产同等级的产品&#xff0c;但是在现实生活中许多工厂都…

一文读懂代理与反向代理

反向代理与正向代理在代理服务器的角色和工作方式上有所不同&#xff0c;这两种代理方式主要取决于代理服务器代理的对象和处理方式。 区别&#xff1a; 正向代理&#xff1a; 正向代理是代理客户端的请求&#xff0c;客户端通过正向代理访问其他服务或资源。客户端知道自己正…

注解总结,Java中的注解,springboot中的注解

注解总结 1、Junit 开始执行的方法&#xff1a;初始化资源&#xff0c;执行完之后的方法&#xff1a;释放资源 测试方法&#xff0c;必须是&#xff1a;公有、非静态、无参无返回值的 在一个类中&#xff0c;可以定义多个测试方法&#xff0c;每个测试方法可以单独运行&#…

Qt/C++通用跨平台Onvif工具/支持海康大华宇视华为天地伟业等/云台控制/预置位管理/工程调试利器

一、前言 在安防视频监控行业&#xff0c;Onvif作为国际标准&#xff0c;几乎主要的厂商都支持&#xff0c;不仅包含了国内的厂商&#xff0c;也包括主要的国际厂商&#xff0c;由于有了这个标准的存在&#xff0c;使得不同设备不同安防平台之间&#xff0c;能够接入各个厂家的…

堆(数据结构)

堆的概念及结构 如果有一个关键码的集合K { &#xff0c; &#xff0c; &#xff0c;…&#xff0c; }&#xff0c;把它的所有元素按完全二叉树的顺序存储方式存储在一个一维数组中&#xff0c;并满足&#xff1a; < 且 < ( > 且 > ) i 0&#xff0c;1&#xff…

Linux技巧|Awk 比较运算符

在处理一行文本中的数字或字符串值时&#xff0c;使用比较运算符过滤文本或字符串对于 Awk 命令来说非常方便。 在 Awk 系列的这一部分中&#xff0c;我们将了解如何使用比较运算符过滤文本或字符串。 比较运算符 Awk 中的比较运算符用于比较数字或字符串的值&#xff0c;它们包…

Sticks POJ 1011

题目链接:1011 -- Sticks 题目描述: 评注: 好暴力的一个题&#xff0c;就当锻炼自己的搜索技巧了,本题中的一些剪枝技巧还蛮有意思的&#xff0c;可以品味一下 思路: 枚举深搜剪枝 看完题面&#xff0c;其实感觉没啥特别巧妙的性质&#xff0c;而且数据范围比较小&#xff0…

双指针(滑动窗口)-算法刷题

一.移动零&#xff08;. - 力扣&#xff08;LeetCode&#xff09;&#xff09; 算法思想 &#xff1a; 设置两个指针left,right&#xff0c;将数组分为三块[0,left]为不为0的元素&#xff0c;[left1,right-1]为0元素&#xff0c;[right,num.size()-1]为未扫描的区域&#xff0c…

【动态规划】Leetcode 746. 使用最小花费爬楼梯

【动态规划】Leetcode 746. 使用最小花费爬楼梯 解法 ---------------&#x1f388;&#x1f388;题目链接&#x1f388;&#x1f388;------------------- 解法 &#x1f612;: 我的代码实现> 动规五部曲 ✒️确定dp数组以及下标的含义 dp[i] 表示跳跃到第 i 层&#x…

10一维数组

一维数组是什么&#xff1f; // 相同的数据类型的元素组成的有序集合 // 创建一个一维数组 // 例如&#xff1a; int a[5]{1,2,3,4,5}; //说明&#xff1a; // int:表示数据类型 // a:表示数组名 // [5]&#xff1a;数组的长度 // {1,2,3,4,5} &#xff1a;数组的元素 // 一…

适配器模式与桥接模式-灵活应对变化的两种设计策略大比拼

​&#x1f308; 个人主页&#xff1a;danci_ &#x1f525; 系列专栏&#xff1a;《设计模式》 &#x1f4aa;&#x1f3fb; 制定明确可量化的目标&#xff0c;坚持默默的做事。 &#x1f680; 转载自&#xff1a;设计模式深度解析&#xff1a;适配器模式与桥接模式-灵活应对变…

SSD268G系列高性价比DIS CAM方案

技术详情 星宸科技&#xff08;Sigmastar&#xff09;高端智能显示芯片SSD268G&#xff0c;适用于高分辨率的视频录制和播放应用。作为一款高集成度&#xff0c;多功能的高性能处理器&#xff0c;其内置ARM Cortex-A53双核64位处理器&#xff0c;最高工作主频为1.2GHz&#xf…

下载安装anaconda和pytorch的详细方法,以及遇到的问题和解决办法

下载安装Anaconda 首先需要下载Anaconda&#xff0c;可以到官网Anaconda官网或者这里提供一个镜像网站去下载anaconda镜像网站 安装步骤可参考该文章&#xff1a;Anaconda安装步骤&#xff0c;本篇不再赘述 注意环境变量的配置&#xff0c;安装好Anaconda之后一定要在环境变量…

初始Java篇(JavaSE基础语法)(2)(逻辑控制)

个人主页&#xff08;找往期文章包括但不限于本期文章中不懂的知识点&#xff09;&#xff1a;我要学编程(ಥ_ಥ)-CSDN博客 目录 逻辑控制 顺序结构 分支结构 if语句 switch 语句 循环结构 while 循环 for 循环 do while 循环 输入输出 输出到控制台 从键盘输入 …

<REAL-TIME TRAFFIC OBJECT DETCTION FOR AUTONOMOUS DRIVING>论文阅读

Abstract 随着计算机视觉的最新进展&#xff0c;自动驾驶迟早成为现代社会的一部分&#xff0c;然而&#xff0c;仍有大量的问题需要解决。尽管现代计算机视觉技术展现了优越的性能&#xff0c;他们倾向于将精度优先于效率&#xff0c;这是实时应用的一个重要方面。大型目标检测…

使用git+ssh访问github,避免下载资源失败

一、创建github账户之后&#xff0c;记住注册邮箱和账户名 我的邮箱&#xff1a;yuanyan23mails.ucas.ac.cn 账户名&#xff1a;thekingofjumpshoot 下边的相关位置需要用自己的邮箱和用户名替代 二、输入本地生成秘钥和公钥命令&#xff0c;并且生成公私钥对 ssh-keygen …

WooCommerce Appointments: 建立会话或预订类型网站的详细教程- US Domain Center主机

第一步&#xff1a;了解会话或预订类型网站 会话或预订类型网站是一种让用户可以预订会议、咨询、活动或其他服务的网站。这种类型的网站通常使用预订插件来管理日程安排、接受预订、处理付款等功能。 第二步&#xff1a;注册 US Domain Center 账户 美国域名中心US Domain …