JetPack之ViewModel+LiveData

news2025/1/16 3:38:31

目录

  • 一、概述
  • 二、LiveData 使用
    • 2.1 创建 LiveData 对象
    • 2.2 观察 LiveData 对象
    • 2.3 更新 LiveData 对象
  • 三、编写 LiveData Demo
    • 3.1 不使用 LiveData
    • 3.2 使用 MutableLiveData
    • 3.3 使用 MediatorLiveData
      • 3.3.1 监听 2 个数据源的变化
      • 3.3.2 编写模拟 2 个数据源更新的代码
  • 四、ViewModel 使用
    • 4.1 创建 ViewModel 类
    • 4.2 在 Activity 中使用
  • 五、ViewModel+LiveData
    • 5.1 在 ViewModel 中增加 LiveData
    • 5.2 在 ViewModel 中添加操作数据的逻辑
    • 5.3 修改 MainActivity 中和 liveData 有关的代码
    • 5.4 修改 MyViewModel
    • 5.5 修改后
      • 5.5.1 修改后MainActivity
      • 5.5.2 修改后MyViewModel
  • 六、总结


一、概述

之前单独介绍过ViewModel和LiveData,但是没有将两者结合使用。
JetPack之ViewModel
JetPack之LiveData
ViewModel 对象为特定的界面组件(如 Fragment 或 Activity)提供数据,并包含数据处理业务逻辑,会配合 LiveData 一起使用。
LiveData 是一种可观察的数据存储器类, LiveData 使用观察者模式,每当数据发生变化时, LiveData 会通知 Observer 对象,可以在这些 Observer 对象中更新 UI。
接下来,先介绍如果使用 LiveData,并编写一个 LiveData Demo,接着再结合ViewModel 对 LiveData Demo 的代码进行重构


二、LiveData 使用

LiveData< T>是一个抽象类,它有 2 个子类分别是: MutableLiveData和MediatorLiveData< T>,在编写代码时,是创建的子类。先来看MutableLiveData< T>使用方法,在后面的示例中再介绍如何使用 MediatorLiveData< T>

2.1 创建 LiveData 对象

如果要观察的对象类为 String,就通过如下代码创建一个 MutableLiveData 对象

MutableLiveData< String> liveData = new MutableLiveData<>();

2.2 观察 LiveData 对象

通过 observe 方法来监听 LiveData 的数据变化,每当 LiveData 发生变化时,都会回调
onChanged 方法,并返回值的内容 String s

liveData.observe(this, new Observer< String>() {
@Override
public void onChanged(String s) {
Log.i(TAG, "onChanged: " + s)
   	}
});

2.3 更新 LiveData 对象

通过 LiveData 的 setValue 方法,给 LiveData 赋值,每次赋完值后,都会回调 onChanged 方法

liveData.setValue("add for test");

当给 LiveData 赋值为"add for test"时, onChanged 会输出日志:

I/LiveDataDemo: onChanged: add for test

三、编写 LiveData Demo

创建一个 Demo,主界面只有一个 TextView, TextView 用作展示一个数字,这个数字会从 59 开始显示,然后每隔 1 秒数字会减 1,直到数字变为 0。

3.1 不使用 LiveData

代码说明:
1、在 MainActivity 增加一个 countDown 的方法,通过 CountDownTimer 创建一个倒计时器
2、 new CountDownTimer(1 * 60 * 1000, 1 * 1000)中 2 个参数的含义:第一个参数表示这个 Timer 的总时长为 60 秒;第 2 个参数表示每隔 1 秒中会更新一次,并回调 onTick方法
3、每次调用 onTick 方法,会调用 textView.setText(String.valueOf(l / 1000));设置一次TextView 的内容
对应代码如下:

public class MainActivity extends AppCompatActivity {

    private TextView textView;

    @SuppressLint("MissingInflatedId")
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        textView = findViewById(R.id.textView);
        countDown();
    }

    private void countDown() {
        new CountDownTimer(1 * 60 * 1000, 1 * 1000) {
            @Override
            public void onTick(long l) {
                textView.setText(String.valueOf(l / 1000));
            }

            @Override
            public void onFinish() {
            }
        }.start();
    }
}

布局文件:

<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout 
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".MainActivity">

    <TextView
        android:id="@+id/textView"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        app:layout_constraintBottom_toBottomOf="parent"
        app:layout_constraintEnd_toEndOf="parent"
        app:layout_constraintStart_toStartOf="parent"
        app:layout_constraintTop_toTopOf="parent" />

</androidx.constraintlayout.widget.ConstraintLayout>

效果:
在这里插入图片描述
弊端:
业务逻辑(倒计时的功能)和 UI 逻辑(TextView 更新)没有分开,耦合到一起了

3.2 使用 MutableLiveData

为了方便演示,直接在 MainActivity 中操作 LiveData,实际项目中不要这样编写代码,应该把LiveData 放到 ViewModel 中
Google 推荐的使用 LiveData 的方法:
在这里插入图片描述

代码说明:
1、在 MainActivity 中增加一个 long 类型的 MutableLiveData
2、每次回调 onTick 方法时,调用 liveData.setValue(l);,把最新的值设置给 LiveData
3、调用 LivewData 的 observe 方法,设置一个监听器,每当 LiveData 的值变化时,会回调 onChanged 方法,在 onChanged 中设置 TextView 的内容
对应代码如下:

public class MainActivity extends AppCompatActivity {

    private final MutableLiveData<Long> liveData = new MutableLiveData<>();
    private TextView textView;

    @SuppressLint("MissingInflatedId")
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        textView = findViewById(R.id.textView);
        countDown();
        // TODO:为了方便演示,直接在 MainActivity 中操作 LiveData,实际项目中不要这样编写代码,应该把 LiveData 放到 ViewModel 中
        liveData.observe(this, new Observer<Long>() {
            @Override
            public void onChanged(Long aLong) {
                textView.setText(String.valueOf(aLong / 1000));
            }
        });
        countDown();
    }


    public void countDown() {
        new CountDownTimer(1 * 60 * 1000, 1 * 1000) {
            @Override
            public void onTick(long l) {
                // TODO:为了方便演示,直接在 MainActivity 中操作 LiveData,实际项目中不要这样编写代码,应该把 LiveData 放到 ViewModel 中
                liveData.setValue(l);
            }

            @Override
            public void onFinish() {
            }
        }.start();
    }

}

效果与上面一样。

3.3 使用 MediatorLiveData

MediatorLiveData 可以合并多个 LiveData 源,只要任何原始的 LiveData 源对象发生更改,就会触发 MediatorLiveData 对象的观察者。
例如,如果界面中有可以从本地数据库或网络更新的 LiveData 对象,则可以向MediatorLiveData 对象添加以下源:

与存储在数据库中的数据关联的LiveData 对象。
与从网络访问的数据关联的LiveData 对象。

Activity 只需观察 MediatorLiveData 对象即可从这两个源接收更新。接下来,实现这个例子。

3.3.1 监听 2 个数据源的变化

代码说明如下:
1、创建 1 个 MediatorLiveData< String>对象
2、创建 2 个MutableLiveData: liveData1 表示网络数据源, liveData2 表示本地数据源
3、调用 MediatorLiveData 的 addSource 方法,分别将 liveData1 和 liveData2 加到 MediatorLiveData 要监听的数据源中
4、设置 MediatorLiveData 的监听,当 2 个数据源发生变化时,会回调onChanged 方法,并将变化的内容展示到 TextView 上

3.3.2 编写模拟 2 个数据源更新的代码

编写一个 mergeTes 的方法,里面创建了 2 个 Timer,用作模拟网络数据更新和本地数据更新的情况,每隔 3 秒会往网络数据 liveData1 设置值、每隔 10 秒会往本地数据liveData2 设置值。
完整代码如下

public class MainActivity extends AppCompatActivity {

    private TextView textView;
    private MutableLiveData< String> liveData1 = new MutableLiveData<>();
    private MutableLiveData< String> liveData2 = new MutableLiveData<>();
    final MediatorLiveData< String> liveDataMerger = new MediatorLiveData<>();
    @SuppressLint("MissingInflatedId")
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        textView = findViewById(R.id.textView);

        liveDataMerger.addSource(liveData1, new Observer< String>() {
            @Override
            public void onChanged(String s) {
                liveDataMerger.setValue(s);
            }
        });
        liveDataMerger.addSource(liveData2, new Observer< String>() {
            @Override
            public void onChanged(String s) {
                liveDataMerger.setValue(s);
            }});
        liveDataMerger.observe(this, new Observer< String>() {
            @Override
            public void onChanged(String s) {
                textView.setText(s);
            }
        });
        mergeTest();
    }



    public void mergeTest() {
        new CountDownTimer(1 * 60 * 1000, 3 * 1000) {
            @Override
            public void onTick(long l) {
                liveData1.setValue("网络有数据更新了" + l/1000);
            }
            @Override
            public void onFinish() {
            }
        }.start();
        new CountDownTimer(1 * 60 * 1000, 10 * 1000) {
            @Override
            public void onTick(long l) {
                liveData2.setValue("本地数据库更新了" + l/1000);
            }
            @Override
            public void onFinish() {
            }
        }.start();
    }
}

运行效果

在这里插入图片描述

四、ViewModel 使用

4.1 创建 ViewModel 类

创建一个 MyViewModel 类,继承自 ViewModel

import androidx.lifecycle.ViewModel;
public class MyViewMode extends ViewModel {

}

4.2 在 Activity 中使用

    public class MainActivity extends AppCompatActivity {
        private MyViewModel viewModel;
        @Overrideprotected
        void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_main);
            viewModel = new ViewModelProvider(this).get(MyViewModel.class);
        }
    }

五、ViewModel+LiveData

5.1 在 ViewModel 中增加 LiveData

在 MyViewModel 中增加 3 个 MutableLiveData,分别对应在 MainActivity 中使用到的。
并编写这 3 个 LiveData 的 get 方法

public class MyViewModel extends ViewModel {

    private MutableLiveData<Long> liveData = new MutableLiveData<>();
    private MutableLiveData<String> liveData1 = new MutableLiveData<>();
    private MutableLiveData<String> liveData2 = new MutableLiveData<>();

    public MutableLiveData<Long> getLiveData() {
        return liveData;
    }

    public MutableLiveData<String> getLiveData1() {
        return liveData1;
    }

    public MutableLiveData<String> getLiveData2() {
        return liveData2;
    }

}

5.2 在 ViewModel 中添加操作数据的逻辑

countDown、 mergeTest 这 2 个方法在操作数据,从 MainActivity 中把这 2 个方法copy 到 MyViewModel 中

    public void countDown() {
        new CountDownTimer(1 * 60 * 1000, 1 * 1000) {
            @Override
            public void onTick(long l) {

                liveData.postValue(l);
            }

            @Override
            public void onFinish() {
            }
        }.start();
    }

    public void mergeTest() {
        new CountDownTimer(1 * 60 * 1000, 3 * 1000) {
            @Override
            public void onTick(long l) {
                liveData1.postValue("网络有数据更新了" + l / 1000);
            }

            @Override
            public void onFinish() {
            }
        }.start();
        new CountDownTimer(1 * 60 * 1000, 10 * 1000) {
            @Override
            public void onTick(long l) {
                liveData2.postValue("本地数据库更新了" + l / 1000);
            }

            @Override
            public void onFinish() {
            }
        }.start();
    }

5.3 修改 MainActivity 中和 liveData 有关的代码

1、删除 countDown,改用 viewModel.countDown();

    MyViewModel viewModel = new ViewModelProvider(this).get(MyViewModel.class);
    viewModel.countDown();

2、用 viewModel.getLiveData().observe 替换 liveData.observe

    viewModel.getLiveData().observe(this, new Observer<Long>() {
        @Override
        public void onChanged(Long aLong) {
            textView.setText(String.valueOf(aLong / 1000));
        }
    });

5.4 修改 MyViewModel

增加 liveDataMerger 字段,并编写 getLiveDataMerger()方法,这个方法里面的内容,是从 MainActivity 中 copy 过来的

public class MyViewModel extends ViewModel {

    private MediatorLiveData<String> liveDataMerger;

    private MutableLiveData<Long> liveData = new MutableLiveData<>();
    private MutableLiveData<String> liveData1 = new MutableLiveData<>();
    private MutableLiveData<String> liveData2 = new MutableLiveData<>();

    public MediatorLiveData<String> getLiveDataMerger() {
        if (liveDataMerger == null) {
            liveDataMerger = new MediatorLiveData<>();
        }
        liveDataMerger.addSource(liveData1, new Observer<String>() {
            @Override
            public void onChanged(String s) {

                liveDataMerger.setValue(s);
            }
        });
        liveDataMerger.addSource(liveData2, new Observer<String>() {
            @Override
            public void onChanged(String s) {

                liveDataMerger.setValue(s);
            }
        });
        return liveDataMerger;
    }
.........

5.5 修改后

5.5.1 修改后MainActivity

public class MainActivity extends AppCompatActivity {

    private TextView textView;
    private MyViewModel viewModel;

    @SuppressLint("MissingInflatedId")
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        textView = findViewById(R.id.textView);
        viewModel = new ViewModelProvider(this).get(MyViewModel.class);

        viewModel.getLiveDataMerger().observe(this, new Observer<String>() {
            @Override
            public void onChanged(String s) {
                textView.setText(s);
            }
        });
        viewModel.mergeTest();
//        viewModel.getLiveData().observe(this, new Observer<Long>() {
//            @Override
//            public void onChanged(Long aLong) {
//                textView.setText(String.valueOf(aLong / 1000));
//            }
//        });
//        viewModel.countDown();
    }
}

5.5.2 修改后MyViewModel

public class MyViewModel extends ViewModel {
    private MutableLiveData<Long> liveData = new MutableLiveData<>();
    private MutableLiveData<String> liveData1 = new MutableLiveData<>();
    private MutableLiveData<String> liveData2 = new MutableLiveData<>();
    private MediatorLiveData<String> liveDataMerger;

    public MediatorLiveData<String> getLiveDataMerger() {
        if (liveDataMerger == null) {
            liveDataMerger = new MediatorLiveData<>();
        }
        liveDataMerger.addSource(liveData1, new Observer<String>() {
            @Override
            public void onChanged(String s) {
                liveDataMerger.setValue(s);
            }
        });
        liveDataMerger.addSource(liveData2, new Observer<String>() {
            @Override
            public void onChanged(String s) {

                liveDataMerger.setValue(s);
            }
        });
        return liveDataMerger;
    }

    public MutableLiveData<Long> getLiveData() {
        return liveData;
    }

    public MutableLiveData<String> getLiveData1() {
        return liveData1;
    }

    public MutableLiveData<String> getLiveData2() {
        return liveData2;
    }

    public void countDown() {
        new CountDownTimer(1 * 60 * 1000, 1 * 1000) {
            @Override
            public void onTick(long l) {

                liveData.postValue(l);
            }

            @Override
            public void onFinish() {
            }
        }.start();
    }

    public void mergeTest() {
        new CountDownTimer(1 * 60 * 1000, 3 * 1000) {
            @Override
            public void onTick(long l) {
                liveData1.postValue("网络有数据更新了" + l / 1000);
            }

            @Override
            public void onFinish() {
            }
        }.start();
        new CountDownTimer(1 * 60 * 1000, 10 * 1000) {
            @Override
            public void onTick(long l) {
                liveData2.postValue("本地数据库更新了" + l / 1000);
            }

            @Override
            public void onFinish() {
            }
        }.start();
    }
}

六、总结

ViewModel 和 LiveData 结合使用的一些好处:

  • 生命周期感知:ViewModel 和 LiveData 都是生命周期感知的组件,它们可以在界面控制器(如 Activity 或 Fragment)的生命周期变化时自动调整其行为。这意味着可以在 ViewModel 中存储和管理数据,而 LiveData 可以通知观察者(如界面控制器)数据的变化。
  • 数据持久性:ViewModel 和 LiveData 结合使用可以确保数据在配置更改(如屏幕旋转)时得以保留。ViewModel 会在界面控制器销毁和重新创建时保持数据不变,而 LiveData 可以通知观察者更新数据。
  • 避免内存泄漏:ViewModel 和 LiveData 的结合使用有助于避免内存泄漏问题。ViewModel 不持有对界面控制器的引用,而 LiveData 会自动处理观察者的生命周期,从而避免因观察者未及时取消注册而导致的内存泄漏。
  • 数据更新通知:LiveData 可以实现数据的观察和更新,当数据发生变化时,LiveData 会通知所有观察者进行相应的更新操作。这样可以确保界面与数据保持同步,提升用户体验。
  • 简化异步操作:LiveData 可以结合 ViewModel 使用,帮助简化异步操作和数据加载过程。可以在 ViewModel 中处理异步操作,然后通过 LiveData 将结果传递给观察者,从而实现响应式编程。
  • 解耦数据和界面:ViewModel 和 LiveData 的结合使用有助于将数据逻辑与用户界面分离,使代码更易于维护和测试。ViewModel 中的数据可以供多个界面控制器共享,避免重复加载数据。

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

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

相关文章

4月威胁态势 | 0day占比82%!Polyransom勒索家族强势来袭

近日&#xff0c;亚信安全正式发布《2024年4月威胁态势报告》&#xff08;以下简称“报告”&#xff09;&#xff0c;报告显示&#xff0c;4月份新增安全漏洞1260个&#xff0c;涉及0day漏洞占82%&#xff1b;监测发现当前较活跃的勒索病毒家族是Polyransom和Blocker&#xff0…

FIFO Generate IP核使用——Native接口时读写宽度不一致详解

表3-4显示了在选择了Native接口时哪些时钟类型支持非对称的宽高比&#xff0c;即读写宽度不一致。 非对称的宽高比允许FIFO的输入和输出宽度不同。FIFO支持以下写-读宽高比&#xff08;Write-to-Read Aspect Ratios&#xff09;&#xff1a;1:8, 1:4, 1:2, 1:1, 2:1, 4:1, 8:1…

【数据分析】这些年我发过的微信朋友圈

TencentRecordAnalysisV1.0.3.zip 蓝奏云&#xff1a;链接:链接TencentRecordAnalysis (lanzoub.com)密码:9hww 朋友圈还是以本行业岩土、工作相关的内容居多。 对于一个不怎么发圈的人来说&#xff0c;这几天有点反常&#xff0c;这几天大概是我成功的开发了几个失败的GPT应用…

C++:set和map的介绍

目录 关联式容器 键值对 set介绍&#xff1a; set的模板参数列表 set的双向迭代器&#xff1a; insert的使用和set的特性&#xff1a; set的删除&#xff1a; set的find&#xff1a; lower_bound 、 upper_bound&#xff1a; multiset&#xff1a; map介绍&#xff…

机器学习-K近邻算法(KNN)

目录 什么是KNN算法 图解KNN基本算法 &#xff08;1&#xff09;k近邻算法中k的选取 &#xff08;2&#xff09;距离函数 &#xff08;3&#xff09;归一化处理 &#xff08;4&#xff09;概率kNN KNN算法的优缺点 优势 缺点 KNN算法总结 什么是KNN算法 k近邻算法&…

MySQL-集群的高可用

MMM: Multi-Master Replication Manager for MySQL&#xff0c;Mysql主主复制管理器是一套灵活的脚本程序&#xff0c;基于perl实现&#xff0c;用来对mysql replication进行监控和故障迁移&#xff0c;并能管理mysql Master-Master复制的配置(同一时间只有一个节点是可写的) …

成功者的8个时间管理技巧,让你在职场中脱颖而出

在职场中&#xff0c;高效的时间管理不仅能帮助你完成更多工作&#xff0c;还能提升你的工作效率和工作质量。以下是八个时间管理技巧&#xff1a; 一、明确职业目标&#xff0c;设定优先级 明确自己的职业目标&#xff0c;并根据这些目标设定工作任务的优先级。这有助于你专…

HTML5实现酷炫个人产品推广、工具推广、信息推广、个人主页、个人介绍、酷炫官网、门户网站模板源码

文章目录 1.设计来源1.1 主界面1.2 我的产品界面1.3 关于我们界面1.4 照片墙界面1.5 发展历程界面1.6 优秀人才界面1.7 热门产品界面1.8 联系我们界面 2.灵活调整模块3.效果和源码3.1 动态效果3.2 源代码 源码下载 作者&#xff1a;xcLeigh 文章地址&#xff1a;https://blog.c…

PADS 规则设置-导线不跟随器件-导线允许回路

1、PADS Layout中设置拖动器件时导线不跟着移动 2、PADS Router中设置走线允许回路

Python基础学习之logging模块

在Python编程中&#xff0c;日志记录&#xff08;Logging&#xff09;是一个非常重要的功能。它不仅可以帮助我们追踪和调试代码中的错误&#xff0c;还可以记录程序运行时的关键信息&#xff0c;以便后续分析和优化。Python标准库中的logging模块为我们提供了强大的日志记录功…

受控文件传输要怎么做 才能保障数据不泄露?

受控文件传输是指在严格的安全措施和政策控制下进行文件传输的过程&#xff0c;受控文件传输的目的是确保数据的安全性和完整性&#xff0c;同时满足合规性和审计要求&#xff0c;适用于需要高安全级别的企业环境&#xff0c;如金融机构、医疗机构、政府机构等。 为了实现受控文…

Rust 实战thiserror+自定义错误消息体

导航 一、背景二、实践1、导入thiserror2、自定义错误消息体&#xff08;1&#xff09;创建ErrMsg.rs和创建自定义结构体&#xff08;2&#xff09;lib.rs添加ErrMsg&#xff08;3&#xff09;main函数&#xff08;4&#xff09;完整代码 一、背景 开发中遇到需要通用、能够满…

发卡授权盗u源码系统搭建ZHU16728

2024最新UI发卡盗U/支持多语言/更新UI界面/支持多个主流钱包去除后门板&#xff0c;搭建系统TGaqxm01 。功能完美。 此套系统二开后完美授权成功&#xff0c;百分百授权。不是市面上那种授不了权的那种。 授权额度Unlimited 懂的都懂。 1.Php静态 2.目录puicta 3.扩sal 4.ss…

避雷!7.7分,新增1区TOP被标记On Hold,5本已被踢除!

本周投稿推荐 SSCI • 2/4区经管类&#xff0c;2.5-3.0&#xff08;录用率99%&#xff09; SCIE&#xff08;CCF推荐&#xff09; • 计算机类&#xff0c;2.0-3.0&#xff08;最快18天录用&#xff09; SCIE&#xff08;CCF-C类&#xff09; • IEEE旗下&#xff0c;1/2…

Mysql索引失效情况

索引失效的情况 这是正常查询情况&#xff0c;满足最左前缀&#xff0c;先查有先度高的索引。 1. 注意这里最后一种情况&#xff0c;这里和上面只查询 name 小米科技 的命中情况一样。说明索引部分丢失&#xff01; 2. 这里第二条sql中的&#xff0c;status > 1 就是范围查…

【图书推荐】《ChatGLM3大模型本地化部署、应用开发与微调》

本书目的 本书通过多个PyTorch实战案例&#xff0c;帮助读者掌握ChatGLM3大模型本地化部署、应用开发与微调技能。智能问答机器人、美妆助手、上市公司财务报表信息抽取、上市公司财务报表智能问答与财务预警等案例&#xff0c;都可以按读者自己的业务需求&#xff0c;改造成可…

Ubuntu多版本(低版本)gcc/g++安装、切换与卸载图文教程

目录 1 问题背景2 多版本安装3 多版本切换4 多版本卸载5 其他问题 1 问题背景 环境&#xff1a; gcc 9.4.0g 9.4.0Ubuntu20.04 现象&#xff1a;通过apt install build-essential安装的gcc和g默认是当前版本系统支持的最高版本编译器&#xff0c;但是很多工程的编译需要安装低版…

大数据信用风险特别高,那大数据信用高风险要如何降低呢?

在大数据信用报告中&#xff0c;综合评分是直观体现信用风险高低的重要元素&#xff0c;也就是我们长听说的大数据信用分&#xff0c;很多人在查大数据信用报告之后&#xff0c;发现自己的大数据信用风险特别高&#xff0c;那大数据信用高风险要如何降低呢?小编从引起高风险的…

在线点餐系统搭建重点,会用到哪些三方服务?

在线点餐系统的搭建是一个涉及多个环节和技术的过程&#xff0c;其重点包括需求分析、系统设计、功能实现、用户体验以及系统的稳定性和安全性等方面。在搭建过程中&#xff0c;可能会使用到以下三方服务&#xff1a; 云服务提供商&#xff1a;云服务为在线点餐系统提供了基础…