MVVM(Model-View-ViewModel)架构模式
在Android开发中,MVVM(Model-View-ViewModel)架构模式已经成为构建可维护和可扩展应用程序的重要选择。MVVM模式通过分离视图(View)、模型(Model)和视图模型(ViewModel)来提高代码的可读性、可测试性和可维护性。本文将详细探讨MVVM模式在Android架构组件中的实战应用,通过具体示例来说明如何搭建和实现MVVM架构。
一、MVVM架构概述
MVVM模式由三部分组成:Model(模型)、View(视图)和ViewModel(视图模型)。在Android开发中,这三部分各自承担着不同的职责:
- Model:承载业务逻辑与数据实体,独立于UI并与ViewModel进行交互,实现数据获取与处理功能。
- View:用户界面层,负责显示数据和接收用户输入。在Android中,通常使用XML布局文件和Activity/Fragment等组件来创建视图。
- ViewModel:作为View与Model之间的桥梁,封装UI状态与逻辑,具备生命周期感知特性,确保在配置改变时数据得以持久保存。
二、实战应用步骤
1. 准备工作
在开始MVVM架构的实战应用之前,需要准备一些必要的依赖库和工具。Android Jetpack提供了许多有助于实现MVVM架构的组件,如LiveData、ViewModel和Room等。此外,还可以引入Dagger2进行依赖注入,RxJava进行反应式编程,以及FastAndroidNetworking进行网络请求。
添加依赖
在项目的build.gradle
文件中添加以下依赖:
dependencies {// ViewModel和LiveDataimplementation "androidx.lifecycle:lifecycle-viewmodel-ktx:2.4.0"implementation "androidx.lifecycle:lifecycle-livedata-ktx:2.4.0"// Room数据库implementation "androidx.room:room-runtime:2.4.0"annotationProcessor "androidx.room:room-compiler:2.4.0"// Dagger2implementation 'com.google.dagger:dagger:2.38.1'annotationProcessor 'com.google.dagger:dagger-compiler:2.38.1'// RxJavaimplementation 'io.reactivex.rxjava3:rxjava:3.1.2'// FastAndroidNetworkingimplementation 'com.amitshekhar.android:android-networking:1.0.2'// 其他必要依赖...
}
2. 创建Model层
Model层负责数据管理和业务逻辑处理。在Android中,可以使用Java类来定义数据模型,或者使用Room数据库来管理持久化数据。
示例数据模型
public class User {private String name;private int age;// 构造函数、Getter和Setter方法public User(String name, int age) {this.name = name;this.age = age;}// Getter和Setter省略...
}
Room数据库
如果需要持久化存储数据,可以使用Room。首先定义一个Dao接口,然后创建数据库类。
@Dao
public interface UserDao {@Insertvoid insert(User user);@Query("SELECT * FROM user_table")LiveData<List<User>> getAllUsers();
}@Database(entities = {User.class}, version = 1, exportSchema = false)
public abstract class AppDatabase extends RoomDatabase {public abstract UserDao userDao();// 单例获取数据库实例private static volatile AppDatabase INSTANCE;public static AppDatabase getDatabase(Context context) {if (INSTANCE == null) {synchronized (AppDatabase.class) {if (INSTANCE == null) {INSTANCE = Room.databaseBuilder(context.getApplicationContext(),AppDatabase.class, "app_database").build();}}}return INSTANCE;}
}
3. 创建ViewModel层
ViewModel层负责封装UI相关的逻辑和数据,具有生命周期感知能力。通过LiveData,ViewModel可以确保在UI组件的生命周期内更新数据。
示例ViewModel
public class UserViewModel extends ViewModel {private MutableLiveData<User> userLiveData;private UserDao userDao;public UserViewModel(Application application) {AppDatabase db = AppDatabase.getDatabase(application);userDao = db.userDao();userLiveData= new MutableLiveData<>();// 假设我们从数据库加载用户数据loadUser();}private void loadUser() {// 使用LiveData来监听数据库变化userLiveData.setValue(null); // 初始化为null或默认用户对象// 这里使用LiveData从Room获取数据,Room已经处理了线程切换userDao.getAllUsers().observeForever(new Observer<List<User>>() {@Overridepublic void onChanged(List<User> users) {if (!users.isEmpty()) {// 假设我们只关心第一个用户userLiveData.setValue(users.get(0));}// 注意:在实际应用中,可能需要移除这个observeForever监听器,// 或者使用ViewModel的onCleared()方法来处理}});// 注意:上面的代码示例中,直接使用observeForever可能不是最佳实践,// 因为这会导致ViewModel持有LiveData的永久引用,从而可能导致内存泄漏。// 更好的做法是在Fragment或Activity中通过getViewLifecycleOwner().observe()来观察LiveData。// 但为了简单起见,这里使用了observeForever。}public LiveData<User> getUserLiveData() {return userLiveData;}// 还可以添加其他方法来处理用户数据的更新、删除等操作
}
4. 创建View层
View层负责展示数据和接收用户输入。在Android中,这通常是通过Activity或Fragment来实现的,同时结合XML布局文件来定义界面。
示例Fragment
public class UserFragment extends Fragment {private UserViewModel userViewModel;@Overridepublic View onCreateView(LayoutInflater inflater, ViewGroup container,Bundle savedInstanceState) {View view = inflater.inflate(R.layout.fragment_user, container, false);// 获取ViewModel实例userViewModel = ViewModelProviders.of(this).get(UserViewModel.class);// 观察LiveData变化并更新UIuserViewModel.getUserLiveData().observe(getViewLifecycleOwner(), new Observer<User>() {@Overridepublic void onChanged(User user) {// 更新UI,例如显示用户名称和年龄TextView userName = view.findViewById(R.id.user_name);TextView userAge = view.findViewById(R.id.user_age);if (user != null) {userName.setText(user.getName());userAge.setText(String.valueOf(user.getAge()));}}});return view;}
}
注意:从Android Jetpack的androidx.lifecycle:lifecycle-extensions:2.x.x
版本开始,推荐使用ViewModelProvider.Factory
来创建ViewModel的实例,并通过ViewModelProvider.of(this, factory)
来获取ViewModel。但从androidx.lifecycle:lifecycle-viewmodel-ktx:2.x.x
开始,可以直接使用ViewModelProvider.of(this)
来获取ViewModel实例,因为库内部已经处理了ViewModel的创建和缓存逻辑。
5. 总结
通过以上步骤,我们实现了一个简单的MVVM架构在Android中的应用。Model层负责数据管理,ViewModel层负责封装UI逻辑和数据,View层负责展示数据和接收用户输入。这样的架构使得代码更加清晰、易于维护和测试。在实际项目中,还可以结合Dagger2进行依赖注入,RxJava进行反应式编程等高级特性,来进一步提升应用的性能和可维护性。
相关文章:
MVVM(Model-View-ViewModel)架构模式
在Android开发中,MVVM(Model-View-ViewModel)架构模式已经成为构建可维护和可扩展应用程序的重要选择。MVVM模式通过分离视图(View)、模型(Model)和视图模型(ViewModel)来…...

C#MVC返回DataTable到前端展示。
很久没写博客了,闭关太久,失踪人口回归,给诸位道友整点绝活。 交代下背景:要做一个行转列的汇总统计,而且,由于是行转列,列的数量不固定,所以,没法使用正常的SqlSugar框…...
HttpUtils工具类(二)Apache HttpClient 5 使用详细教程
目录 一、Apache HttpClient 5介绍 (1)核心特性 (2)Apache HttpClient 5 的新特性 (3)在 Java 项目的主要使用场景及缺点 使用场景: 缺点: 二、在实际项目中的应用 …...

Vue3.0生命周期钩子(包含:Vue 2.0 和 Vue 3.0)
1、Vue 2.0 生命周期钩子 每个应用程序实例在创建时都有一系列的初始化步骤。例如,创建数据绑定、编译模板、将实例挂载到 DOM 并在数据变化时触发 DOM 更新、销毁实例等。在这个过程中会运行一些叫做生命周期钩子的函数,通过这些钩子函数可以定义业务逻…...

遥感之常用各种指数总结大全
目前在遥感领域基本各种研究领域都会用到各种各样的指数,如水体指数,植被指数,农业长势指数,盐分指数,云指数,阴影指数,建筑物指数,水质指数,干旱指数等等众多。 本文对上…...

【C++】C++11新增特性
目录 C11简介: 1、统一的列表初始化: std::initializer_list 2、自动类型推导: auto: decltype: 3、final 和 override final: override: 4、默认成员函数控制: 显示缺省…...

【LeetCode每日一题】——662.二叉树最大宽度
文章目录 一【题目类别】二【题目难度】三【题目编号】四【题目描述】五【题目示例】六【题目提示】七【解题思路】八【时间频度】九【代码实现】十【提交结果】 一【题目类别】 广度优先搜索 二【题目难度】 中等 三【题目编号】 662.二叉树最大宽度 四【题目描述】 给…...

第二十三节、血量更新逻辑的实现
一、创建代码 引入命名空间 using UnityEngine.UI; 调用UI必须有这个代码 二、ScriptObject类 1、是一个持久化存储文件的类型 接收所有的事件方法 先继承SO类,然后创建项目菜单 2、进行订阅 放入事件类,关联代码,即可进行广播 传递给这…...
Spring Authorization Server 认证服务器搭建
Spring Authorization Server实现了oauth2和oidc,最近有了解相关技术的需求,所以就尝试着进行了基本的环境搭建和技术测试,目前只测试了授权码模式,做一个记录,后续需要用时方便查找和参考。 1. 版本要求 Spring Authorization Server 版本:1.3.1 JDK 版本:17 Spring B…...

秋招突击——8/15——知识补充——垃圾回收机制
文章目录 引言正文指针引用可达性分析算法垃圾回收算法标记清除算法标记整理算法复制分代收集 垃圾收集器Serial收集器ParNew并行收集器Parallel Scavenge吞吐量优先收集器Serial Old老年代收集器Parallel old收集器CMS收集器G1收集器(Garbage First垃圾优先&#x…...

【iOS】UITableViewCell的重用问题解决方法
我自己在实验中对cell的重用总结如下: 非自定义Cell和非自定义cell的复用情况一样: 第一次加载创建tableView的时候,是屏幕上最多也显示几行cell就先创建几个cell,此时复用池里什么都没有开始下滑tableView,刚开始滑…...
开发一个微信小程序商城需要哪些技术栈
开发一个小程序商城需要掌握以下技术栈: 前端技术:包括HTML、CSS和JavaScript,用于定义商城的页面结构、样式设计和交互功能。 微信小程序专用技术:如WXML、WXSS、JavaScript和JSON,用于描述小程…...

望繁信科技荣膺上海市浦东新区博士后创新实践基地称号
近日,上海望繁信科技有限公司(简称“望繁信科技”)凭借在大数据流程智能领域的卓越表现,成功入选上海市浦东新区博士后创新实践基地。这一荣誉不仅是对望繁信科技创新能力和技术实力的高度认可,也标志着公司在推动产学…...

Nginx--代理与负载均衡(扩展nginx配置7层协议及4层协议方法、会话保持)
前言:本博客仅作记录学习使用,部分图片出自网络,如有侵犯您的权益,请联系删除 一、代理原理 1、反向代理产生的背景 单个服务器的处理客户端(用户)请求能力有一个极限,当接入请求过多时&#…...

Ubuntu20.4 系统安装后无wifi图标
0. 问题排查 1.检查 BIOS 设置: 有时候,无线网卡可能在 BIOS 中被禁用。重启电脑,进入 BIOS 设置,确保无线网卡选项是启用的。 2.检查硬件开关: 检查您的笔记本电脑是否有物理开关或键盘快捷键来启用或禁用无线网卡。 3.在软件更新中切换…...

牛客网SQL进阶135 :每个6/7级用户活跃情况
每个67级用户活跃情况_牛客题霸_牛客网 0 问题描述 基于用户信息表user_info、、试卷作答记录表exam_record、题目练习记录表practice_record,统计 每个6/7级用户总活跃月份数、2021年活跃天数、2021年试卷作答活跃天数、2021年答题活跃天数,结果 按照总…...
SQLite3使用接口写入二进制文件
使用接口的方式写入二进制文件 ,有二种方案。 一、全部文件 一次性写下到数据中 使用sqlite3_bind_blob接口 FILE* fpfopen("user.bmp","rb"); iLenfread(buffer,1,65535,fp); fclose(fp);sqlite3_prepare(pDB,"insert into user values …...
在复杂的数据库架构中,如何优化 SQL 查询以提高性能和减少资源消耗?
在优化 SQL 查询以提高性能和减少资源消耗时,可以考虑以下几个方面: 使用索引:为经常被查询的列创建索引,可以大大加快查询速度。同时,避免过多的索引,因为过多的索引会增加写入操作的开销。 编写高效的查…...

【HarmonyOS】端云一体化初始化项目
简介 端云一体化开发是HarmonyOS对云端开发的支持、实现端云联动。云开发服务提供了云函数、云数据库、云存储等服务,可以使开发者专注于应用的业务逻辑开发,无需关注基础设施,例如:服务器、操作系统等问题。 因此,…...
LLM之KG:利用大语言模型(LLM)对文本语料提取概念和概念之间的语义关系进而实现自动构建知识图谱
LLM之KG:利用大语言模型(LLM)对文本语料提取概念和概念之间的语义关系进而实现自动构建知识图谱 目录 ML之KG:基于MovieLens电影评分数据集利用基于知识图谱的推荐算法(networkx+基于路径相似度的方法)实现对用户进行Top电影推荐案例 LLMs之AutoKG:《大型语言模型在知识图…...

利用最小二乘法找圆心和半径
#include <iostream> #include <vector> #include <cmath> #include <Eigen/Dense> // 需安装Eigen库用于矩阵运算 // 定义点结构 struct Point { double x, y; Point(double x_, double y_) : x(x_), y(y_) {} }; // 最小二乘法求圆心和半径 …...

LBE-LEX系列工业语音播放器|预警播报器|喇叭蜂鸣器的上位机配置操作说明
LBE-LEX系列工业语音播放器|预警播报器|喇叭蜂鸣器专为工业环境精心打造,完美适配AGV和无人叉车。同时,集成以太网与语音合成技术,为各类高级系统(如MES、调度系统、库位管理、立库等)提供高效便捷的语音交互体验。 L…...
三维GIS开发cesium智慧地铁教程(5)Cesium相机控制
一、环境搭建 <script src"../cesium1.99/Build/Cesium/Cesium.js"></script> <link rel"stylesheet" href"../cesium1.99/Build/Cesium/Widgets/widgets.css"> 关键配置点: 路径验证:确保相对路径.…...
PHP和Node.js哪个更爽?
先说结论,rust完胜。 php:laravel,swoole,webman,最开始在苏宁的时候写了几年php,当时觉得php真的是世界上最好的语言,因为当初活在舒适圈里,不愿意跳出来,就好比当初活在…...
django filter 统计数量 按属性去重
在Django中,如果你想要根据某个属性对查询集进行去重并统计数量,你可以使用values()方法配合annotate()方法来实现。这里有两种常见的方法来完成这个需求: 方法1:使用annotate()和Count 假设你有一个模型Item,并且你想…...
Java多线程实现之Callable接口深度解析
Java多线程实现之Callable接口深度解析 一、Callable接口概述1.1 接口定义1.2 与Runnable接口的对比1.3 Future接口与FutureTask类 二、Callable接口的基本使用方法2.1 传统方式实现Callable接口2.2 使用Lambda表达式简化Callable实现2.3 使用FutureTask类执行Callable任务 三、…...
【服务器压力测试】本地PC电脑作为服务器运行时出现卡顿和资源紧张(Windows/Linux)
要让本地PC电脑作为服务器运行时出现卡顿和资源紧张的情况,可以通过以下几种方式模拟或触发: 1. 增加CPU负载 运行大量计算密集型任务,例如: 使用多线程循环执行复杂计算(如数学运算、加密解密等)。运行图…...

select、poll、epoll 与 Reactor 模式
在高并发网络编程领域,高效处理大量连接和 I/O 事件是系统性能的关键。select、poll、epoll 作为 I/O 多路复用技术的代表,以及基于它们实现的 Reactor 模式,为开发者提供了强大的工具。本文将深入探讨这些技术的底层原理、优缺点。 一、I…...

RNN避坑指南:从数学推导到LSTM/GRU工业级部署实战流程
本文较长,建议点赞收藏,以免遗失。更多AI大模型应用开发学习视频及资料,尽在聚客AI学院。 本文全面剖析RNN核心原理,深入讲解梯度消失/爆炸问题,并通过LSTM/GRU结构实现解决方案,提供时间序列预测和文本生成…...
探索Selenium:自动化测试的神奇钥匙
目录 一、Selenium 是什么1.1 定义与概念1.2 发展历程1.3 功能概述 二、Selenium 工作原理剖析2.1 架构组成2.2 工作流程2.3 通信机制 三、Selenium 的优势3.1 跨浏览器与平台支持3.2 丰富的语言支持3.3 强大的社区支持 四、Selenium 的应用场景4.1 Web 应用自动化测试4.2 数据…...