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:《大型语言模型在知识图…...
2.Vue编写一个app
1.src中重要的组成 1.1main.ts // 引入createApp用于创建应用 import { createApp } from "vue"; // 引用App根组件 import App from ./App.vue;createApp(App).mount(#app)1.2 App.vue 其中要写三种标签 <template> <!--html--> </template>…...
【SQL学习笔记1】增删改查+多表连接全解析(内附SQL免费在线练习工具)
可以使用Sqliteviz这个网站免费编写sql语句,它能够让用户直接在浏览器内练习SQL的语法,不需要安装任何软件。 链接如下: sqliteviz 注意: 在转写SQL语法时,关键字之间有一个特定的顺序,这个顺序会影响到…...
【Java_EE】Spring MVC
目录 Spring Web MVC 编辑注解 RestController RequestMapping RequestParam RequestParam RequestBody PathVariable RequestPart 参数传递 注意事项 编辑参数重命名 RequestParam 编辑编辑传递集合 RequestParam 传递JSON数据 编辑RequestBody …...
【OSG学习笔记】Day 16: 骨骼动画与蒙皮(osgAnimation)
骨骼动画基础 骨骼动画是 3D 计算机图形中常用的技术,它通过以下两个主要组件实现角色动画。 骨骼系统 (Skeleton):由层级结构的骨头组成,类似于人体骨骼蒙皮 (Mesh Skinning):将模型网格顶点绑定到骨骼上,使骨骼移动…...
USB Over IP专用硬件的5个特点
USB over IP技术通过将USB协议数据封装在标准TCP/IP网络数据包中,从根本上改变了USB连接。这允许客户端通过局域网或广域网远程访问和控制物理连接到服务器的USB设备(如专用硬件设备),从而消除了直接物理连接的需要。USB over IP的…...
Python竞赛环境搭建全攻略
Python环境搭建竞赛技术文章大纲 竞赛背景与意义 竞赛的目的与价值Python在竞赛中的应用场景环境搭建对竞赛效率的影响 竞赛环境需求分析 常见竞赛类型(算法、数据分析、机器学习等)不同竞赛对Python版本及库的要求硬件与操作系统的兼容性问题 Pyth…...
Django RBAC项目后端实战 - 03 DRF权限控制实现
项目背景 在上一篇文章中,我们完成了JWT认证系统的集成。本篇文章将实现基于Redis的RBAC权限控制系统,为系统提供细粒度的权限控制。 开发目标 实现基于Redis的权限缓存机制开发DRF权限控制类实现权限管理API配置权限白名单 前置配置 在开始开发权限…...
深入解析光敏传感技术:嵌入式仿真平台如何重塑电子工程教学
一、光敏传感技术的物理本质与系统级实现挑战 光敏电阻作为经典的光电传感器件,其工作原理根植于半导体材料的光电导效应。当入射光子能量超过材料带隙宽度时,价带电子受激发跃迁至导带,形成电子-空穴对,导致材料电导率显著提升。…...
Java中HashMap底层原理深度解析:从数据结构到红黑树优化
一、HashMap概述与核心特性 HashMap作为Java集合框架中最常用的数据结构之一,是基于哈希表的Map接口非同步实现。它允许使用null键和null值(但只能有一个null键),并且不保证映射顺序的恒久不变。与Hashtable相比,Hash…...
python可视化:俄乌战争时间线关键节点与深层原因
俄乌战争时间线可视化分析:关键节点与深层原因 俄乌战争是21世纪欧洲最具影响力的地缘政治冲突之一,自2022年2月爆发以来已持续超过3年。 本文将通过Python可视化工具,系统分析这场战争的时间线、关键节点及其背后的深层原因,全面…...
