当前位置: 首页 > news >正文

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开发中&#xff0c;MVVM&#xff08;Model-View-ViewModel&#xff09;架构模式已经成为构建可维护和可扩展应用程序的重要选择。MVVM模式通过分离视图&#xff08;View&#xff09;、模型&#xff08;Model&#xff09;和视图模型&#xff08;ViewModel&#xff09;来…...

C#MVC返回DataTable到前端展示。

很久没写博客了&#xff0c;闭关太久&#xff0c;失踪人口回归&#xff0c;给诸位道友整点绝活。 交代下背景&#xff1a;要做一个行转列的汇总统计&#xff0c;而且&#xff0c;由于是行转列&#xff0c;列的数量不固定&#xff0c;所以&#xff0c;没法使用正常的SqlSugar框…...

HttpUtils工具类(二)Apache HttpClient 5 使用详细教程

目录 一、Apache HttpClient 5介绍 &#xff08;1&#xff09;核心特性 &#xff08;2&#xff09;Apache HttpClient 5 的新特性 &#xff08;3&#xff09;在 Java 项目的主要使用场景及缺点 使用场景&#xff1a; 缺点&#xff1a; 二、在实际项目中的应用 &#xf…...

Vue3.0生命周期钩子(包含:Vue 2.0 和 Vue 3.0)

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

遥感之常用各种指数总结大全

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

【C++】C++11新增特性

目录 C11简介&#xff1a; 1、统一的列表初始化&#xff1a; std::initializer_list 2、自动类型推导&#xff1a; auto&#xff1a; decltype&#xff1a; 3、final 和 override final&#xff1a; override&#xff1a; 4、默认成员函数控制&#xff1a; 显示缺省…...

【LeetCode每日一题】——662.二叉树最大宽度

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

第二十三节、血量更新逻辑的实现

一、创建代码 引入命名空间 using UnityEngine.UI; 调用UI必须有这个代码 二、ScriptObject类 1、是一个持久化存储文件的类型 接收所有的事件方法 先继承SO类&#xff0c;然后创建项目菜单 2、进行订阅 放入事件类&#xff0c;关联代码&#xff0c;即可进行广播 传递给这…...

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收集器&#xff08;Garbage First垃圾优先&#x…...

【iOS】UITableViewCell的重用问题解决方法

我自己在实验中对cell的重用总结如下&#xff1a; 非自定义Cell和非自定义cell的复用情况一样&#xff1a; 第一次加载创建tableView的时候&#xff0c;是屏幕上最多也显示几行cell就先创建几个cell&#xff0c;此时复用池里什么都没有开始下滑tableView&#xff0c;刚开始滑…...

开发一个微信小程序商城需要哪些技术栈

开发一个小程序商城需要掌握以下技术栈&#xff1a;‌ 前端技术&#xff1a;‌包括HTML、‌CSS和JavaScript&#xff0c;‌用于定义商城的页面结构、‌样式设计和交互功能。‌ 微信小程序专用技术&#xff1a;‌如WXML、‌WXSS、‌JavaScript和JSON&#xff0c;‌用于描述小程…...

望繁信科技荣膺上海市浦东新区博士后创新实践基地称号

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

Nginx--代理与负载均衡(扩展nginx配置7层协议及4层协议方法、会话保持)

前言&#xff1a;本博客仅作记录学习使用&#xff0c;部分图片出自网络&#xff0c;如有侵犯您的权益&#xff0c;请联系删除 一、代理原理 1、反向代理产生的背景 单个服务器的处理客户端&#xff08;用户&#xff09;请求能力有一个极限&#xff0c;当接入请求过多时&#…...

Ubuntu20.4 系统安装后无wifi图标

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

牛客网SQL进阶135 :每个6/7级用户活跃情况

每个67级用户活跃情况_牛客题霸_牛客网 0 问题描述 基于用户信息表user_info、、试卷作答记录表exam_record、题目练习记录表practice_record&#xff0c;统计 每个6/7级用户总活跃月份数、2021年活跃天数、2021年试卷作答活跃天数、2021年答题活跃天数&#xff0c;结果 按照总…...

SQLite3使用接口写入二进制文件

使用接口的方式写入二进制文件 &#xff0c;有二种方案。 一、全部文件 一次性写下到数据中 使用sqlite3_bind_blob接口 FILE* fpfopen("user.bmp","rb"); iLenfread(buffer,1,65535,fp); fclose(fp);sqlite3_prepare(pDB,"insert into user values …...

在复杂的数据库架构中,如何优化 SQL 查询以提高性能和减少资源消耗?

在优化 SQL 查询以提高性能和减少资源消耗时&#xff0c;可以考虑以下几个方面&#xff1a; 使用索引&#xff1a;为经常被查询的列创建索引&#xff0c;可以大大加快查询速度。同时&#xff0c;避免过多的索引&#xff0c;因为过多的索引会增加写入操作的开销。 编写高效的查…...

【HarmonyOS】端云一体化初始化项目

简介 端云一体化开发是HarmonyOS对云端开发的支持、实现端云联动。云开发服务提供了云函数、云数据库、云存储等服务&#xff0c;可以使开发者专注于应用的业务逻辑开发&#xff0c;无需关注基础设施&#xff0c;例如&#xff1a;服务器、操作系统等问题。 因此&#xff0c;…...

LLM之KG:利用大语言模型(LLM)对文本语料提取概念和概念之间的语义关系进而实现自动构建知识图谱

LLM之KG:利用大语言模型(LLM)对文本语料提取概念和概念之间的语义关系进而实现自动构建知识图谱 目录 ML之KG:基于MovieLens电影评分数据集利用基于知识图谱的推荐算法(networkx+基于路径相似度的方法)实现对用户进行Top电影推荐案例 LLMs之AutoKG:《大型语言模型在知识图…...

C++实现分布式网络通信框架RPC(3)--rpc调用端

目录 一、前言 二、UserServiceRpc_Stub 三、 CallMethod方法的重写 头文件 实现 四、rpc调用端的调用 实现 五、 google::protobuf::RpcController *controller 头文件 实现 六、总结 一、前言 在前边的文章中&#xff0c;我们已经大致实现了rpc服务端的各项功能代…...

【杂谈】-递归进化:人工智能的自我改进与监管挑战

递归进化&#xff1a;人工智能的自我改进与监管挑战 文章目录 递归进化&#xff1a;人工智能的自我改进与监管挑战1、自我改进型人工智能的崛起2、人工智能如何挑战人类监管&#xff1f;3、确保人工智能受控的策略4、人类在人工智能发展中的角色5、平衡自主性与控制力6、总结与…...

在鸿蒙HarmonyOS 5中实现抖音风格的点赞功能

下面我将详细介绍如何使用HarmonyOS SDK在HarmonyOS 5中实现类似抖音的点赞功能&#xff0c;包括动画效果、数据同步和交互优化。 1. 基础点赞功能实现 1.1 创建数据模型 // VideoModel.ets export class VideoModel {id: string "";title: string ""…...

SCAU期末笔记 - 数据分析与数据挖掘题库解析

这门怎么题库答案不全啊日 来简单学一下子来 一、选择题&#xff08;可多选&#xff09; 将原始数据进行集成、变换、维度规约、数值规约是在以下哪个步骤的任务?(C) A. 频繁模式挖掘 B.分类和预测 C.数据预处理 D.数据流挖掘 A. 频繁模式挖掘&#xff1a;专注于发现数据中…...

聊聊 Pulsar:Producer 源码解析

一、前言 Apache Pulsar 是一个企业级的开源分布式消息传递平台&#xff0c;以其高性能、可扩展性和存储计算分离架构在消息队列和流处理领域独树一帜。在 Pulsar 的核心架构中&#xff0c;Producer&#xff08;生产者&#xff09; 是连接客户端应用与消息队列的第一步。生产者…...

python如何将word的doc另存为docx

将 DOCX 文件另存为 DOCX 格式&#xff08;Python 实现&#xff09; 在 Python 中&#xff0c;你可以使用 python-docx 库来操作 Word 文档。不过需要注意的是&#xff0c;.doc 是旧的 Word 格式&#xff0c;而 .docx 是新的基于 XML 的格式。python-docx 只能处理 .docx 格式…...

【Web 进阶篇】优雅的接口设计:统一响应、全局异常处理与参数校验

系列回顾&#xff1a; 在上一篇中&#xff0c;我们成功地为应用集成了数据库&#xff0c;并使用 Spring Data JPA 实现了基本的 CRUD API。我们的应用现在能“记忆”数据了&#xff01;但是&#xff0c;如果你仔细审视那些 API&#xff0c;会发现它们还很“粗糙”&#xff1a;有…...

稳定币的深度剖析与展望

一、引言 在当今数字化浪潮席卷全球的时代&#xff0c;加密货币作为一种新兴的金融现象&#xff0c;正以前所未有的速度改变着我们对传统货币和金融体系的认知。然而&#xff0c;加密货币市场的高度波动性却成为了其广泛应用和普及的一大障碍。在这样的背景下&#xff0c;稳定…...

AI,如何重构理解、匹配与决策?

AI 时代&#xff0c;我们如何理解消费&#xff1f; 作者&#xff5c;王彬 封面&#xff5c;Unplash 人们通过信息理解世界。 曾几何时&#xff0c;PC 与移动互联网重塑了人们的购物路径&#xff1a;信息变得唾手可得&#xff0c;商品决策变得高度依赖内容。 但 AI 时代的来…...

高效线程安全的单例模式:Python 中的懒加载与自定义初始化参数

高效线程安全的单例模式:Python 中的懒加载与自定义初始化参数 在软件开发中,单例模式(Singleton Pattern)是一种常见的设计模式,确保一个类仅有一个实例,并提供一个全局访问点。在多线程环境下,实现单例模式时需要注意线程安全问题,以防止多个线程同时创建实例,导致…...