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:《大型语言模型在知识图…...
Prompt Tuning、P-Tuning、Prefix Tuning的区别
一、Prompt Tuning、P-Tuning、Prefix Tuning的区别 1. Prompt Tuning(提示调优) 核心思想:固定预训练模型参数,仅学习额外的连续提示向量(通常是嵌入层的一部分)。实现方式:在输入文本前添加可训练的连续向量(软提示),模型只更新这些提示参数。优势:参数量少(仅提…...
树莓派超全系列教程文档--(61)树莓派摄像头高级使用方法
树莓派摄像头高级使用方法 配置通过调谐文件来调整相机行为 使用多个摄像头安装 libcam 和 rpicam-apps依赖关系开发包 文章来源: http://raspberry.dns8844.cn/documentation 原文网址 配置 大多数用例自动工作,无需更改相机配置。但是,一…...
Xshell远程连接Kali(默认 | 私钥)Note版
前言:xshell远程连接,私钥连接和常规默认连接 任务一 开启ssh服务 service ssh status //查看ssh服务状态 service ssh start //开启ssh服务 update-rc.d ssh enable //开启自启动ssh服务 任务二 修改配置文件 vi /etc/ssh/ssh_config //第一…...
python/java环境配置
环境变量放一起 python: 1.首先下载Python Python下载地址:Download Python | Python.org downloads ---windows -- 64 2.安装Python 下面两个,然后自定义,全选 可以把前4个选上 3.环境配置 1)搜高级系统设置 2…...
[ICLR 2022]How Much Can CLIP Benefit Vision-and-Language Tasks?
论文网址:pdf 英文是纯手打的!论文原文的summarizing and paraphrasing。可能会出现难以避免的拼写错误和语法错误,若有发现欢迎评论指正!文章偏向于笔记,谨慎食用 目录 1. 心得 2. 论文逐段精读 2.1. Abstract 2…...
1.3 VSCode安装与环境配置
进入网址Visual Studio Code - Code Editing. Redefined下载.deb文件,然后打开终端,进入下载文件夹,键入命令 sudo dpkg -i code_1.100.3-1748872405_amd64.deb 在终端键入命令code即启动vscode 需要安装插件列表 1.Chinese简化 2.ros …...
【android bluetooth 框架分析 04】【bt-framework 层详解 1】【BluetoothProperties介绍】
1. BluetoothProperties介绍 libsysprop/srcs/android/sysprop/BluetoothProperties.sysprop BluetoothProperties.sysprop 是 Android AOSP 中的一种 系统属性定义文件(System Property Definition File),用于声明和管理 Bluetooth 模块相…...
Robots.txt 文件
什么是robots.txt? robots.txt 是一个位于网站根目录下的文本文件(如:https://example.com/robots.txt),它用于指导网络爬虫(如搜索引擎的蜘蛛程序)如何抓取该网站的内容。这个文件遵循 Robots…...
ArcGIS Pro制作水平横向图例+多级标注
今天介绍下载ArcGIS Pro中如何设置水平横向图例。 之前我们介绍了ArcGIS的横向图例制作:ArcGIS横向、多列图例、顺序重排、符号居中、批量更改图例符号等等(ArcGIS出图图例8大技巧),那这次我们看看ArcGIS Pro如何更加快捷的操作。…...
Map相关知识
数据结构 二叉树 二叉树,顾名思义,每个节点最多有两个“叉”,也就是两个子节点,分别是左子 节点和右子节点。不过,二叉树并不要求每个节点都有两个子节点,有的节点只 有左子节点,有的节点只有…...
