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

Android架构组件:MVVM模式的实战应用与数据绑定技巧

目录

引言

一、MVVM模式概述

1.1 MVVM模式简介

1.2 MVVM模式的优势

二、MVVM模式的实现

2.1 项目环境配置

2.2 创建MVVM组件

2.2.1 创建数据模型

2.2.2 创建数据仓库

2.2.3 创建ViewModel

2.2.4 创建布局文件

2.2.5 创建RecyclerView适配器

2.3 在Activity中绑定ViewModel

三、数据绑定技巧

3.1 数据绑定基础

3.2 双向数据绑定

3.3 观察者模式

3.4 使用表达式语言

3.5 性能优化

四、MVVM模式的优缺点分析

优点

缺点

结论


引言

在Android开发中,随着应用复杂度的增加,选择合适的架构模式变得尤为重要。MVVM(Model-View-ViewModel)模式因其清晰的分层结构和高效的开发效率,逐渐成为Android开发者们青睐的架构模式之一。

一、MVVM模式概述

1.1 MVVM模式简介

MVVM是Model-View-ViewModel的缩写,是一种基于数据绑定的架构模式,用于设计和组织应用程序的代码结构。它将应用程序分为三个主要部分:Model(模型)、View(视图)和ViewModel(视图模型)。

  • Model(模型):负责处理数据和业务逻辑。它可以是从网络获取的数据、数据库中的数据或其他数据源。Model层通常是独立于界面的,可以在多个界面之间共享。
  • View(视图):负责展示数据和与用户进行交互。它可以是Activity、Fragment、View等。View层主要负责UI的展示和用户输入的响应。
  • ViewModel(视图模型):连接View和Model,作为View和Model之间的桥梁。它负责从Model中获取数据,并将数据转换为View层可以直接使用的形式。ViewModel还负责监听Model的数据变化,并通知View进行更新。

1.2 MVVM模式的优势

  1. 解耦:通过将UI逻辑与业务逻辑分离,提高了代码的可维护性和测试性。开发者可以独立测试ViewModel和Model,而无需关注View的实现细节。
  2. 数据绑定:Android架构组件提供了数据绑定库,可以简化UI与数据的交互。只需将UI控件与ViewModel中的数据绑定,便可实现双向绑定,从而减少样板代码。
  3. 生命周期感知:使用LiveData与ViewModel结合,能够确保UI在正确的生命周期下进行数据观察,从而避免内存泄露和崩溃。

二、MVVM模式的实现

2.1 项目环境配置

首先,需要在Android项目中引入一些必要的依赖。在build.gradle文件中添加如下依赖:

dependencies {  // ViewModel and LiveData  implementation 'androidx.lifecycle:lifecycle-viewmodel-ktx:2.5.1'  implementation 'androidx.lifecycle:lifecycle-livedata-ktx:2.5.1'  // Data Binding  implementation 'androidx.databinding:databinding-runtime:7.3.1'  // RecyclerView for data display  implementation 'androidx.recyclerview:recyclerview:1.3.1'  
}  android {  ...  buildFeatures {  dataBinding true  }  ...  
}

2.2 创建MVVM组件

接下来,我们将通过一个简单的用户列表应用来展示如何使用MVVM模式。

2.2.1 创建数据模型

首先,定义一个User类来表示用户数据:

data class User(val id: Int, val name: String, val age: Int)

2.2.2 创建数据仓库

创建一个UserRepository类来模拟数据源(比如从网络或本地数据库获取数据):

class UserRepository {  private val users = mutableListOf<User>()  init {  // 初始化一些用户数据  users.add(User(1, "Alice", 25))  users.add(User(2, "Bob", 30))  users.add(User(3, "Charlie", 28))  }  fun getUsers(): List<User> = users  
}

2.2.3 创建ViewModel

ViewModel类用于持有UI相关的数据,并与Model交互。它会暴露一个LiveData对象,这样View可以观察到数据的变化:

import androidx.lifecycle.LiveData  
import androidx.lifecycle.MutableLiveData  
import androidx.lifecycle.ViewModel  class UserViewModel(private val repository: UserRepository) : ViewModel() {  private val _users = MutableLiveData<List<User>>()  val users: LiveData<List<User>> = _users  fun fetchUsers() {  _users.value = repository.getUsers()  }  
}

2.2.4 创建布局文件

在布局文件中启用数据绑定,并使用<data>标签定义ViewModel的绑定变量:

<layout xmlns:android="http://schemas.android.com/apk/res/android">  <data>  <variable  name="viewModel"  type="com.example.mvvmdemo.viewmodel.UserViewModel" />  </data>  <LinearLayout  android:layout_width="match_parent"  android:layout_height="match_parent"  android:orientation="vertical">  <!-- 使用RecyclerView来展示用户数据 -->  <androidx.recyclerview.widget.RecyclerView  android:id="@+id/recyclerView"  android:layout_width="match_parent"  android:layout_height="match_parent"  android:orientation="vertical" />  </LinearLayout>  
</layout>

2.2.5 创建RecyclerView适配器

为了显示用户列表,需要一个RecyclerView适配器。适配器中将使用数据绑定来自动更新数据:

// 假设你已经有一个UserAdapter类实现了RecyclerView.Adapter<UserAdapter.ViewHolder>  
// 并且ViewHolder中使用了数据绑定

2.3 在Activity中绑定ViewModel

MainActivity中进行数据绑定和ViewModel的初始化:

import androidx.activity.viewModels  
import androidx.appcompat.app.AppCompatActivity  
import androidx.databinding.DataBindingUtil  
import com.example.mvvmdemo.databinding.ActivityMainBinding  class MainActivity : AppCompatActivity() {  private lateinit var binding: ActivityMainBinding  private val viewModel: UserViewModel by viewModels()  override fun onCreate(savedInstanceState: Bundle?) {  super.onCreate(savedInstanceState)  binding = DataBindingUtil.setContentView(this, R.layout.activity_main)  binding.viewModel = viewModel  binding.lifecycleOwner = this  // 观察用户数据变化  viewModel.users.observe(this) { users ->  // 更新RecyclerView的适配器  // adapter.submitList(users)  }  // 初始加载用户数据  viewModel.fetchUsers()  }  
}

三、数据绑定技巧

3.1 数据绑定基础

在XML布局文件中,可以直接将ViewModel的属性绑定到视图组件上。例如,使用android:text="@{viewModel.myText}",当ViewModel的myText属性变化时,对应的文本框会被自动更新。

3.2 双向数据绑定

Android数据绑定库支持双向绑定,这意味着UI控件的更改也可以更新数据模型。例如,对于输入框,可以直接绑定到ViewModel的属性,这样在输入框中输入的值会自动更新到ViewModel中:

<EditText  android:id="@+id/editText"  android:layout_width="match_parent"  android:layout_height="wrap_content"  android:hint="Enter name"  android:text="@={viewModel.userName}" />

注意,双向绑定需要使用@={...}语法。

3.3 观察者模式

数据绑定库使用观察者模式来监听数据变化。Observable对象(如LiveData)会在数据变化时发送变更通知,使得View能够响应数据的变化。开发者无需手动编写代码来更新UI,减少了样板代码和潜在的错误。

3.4 使用表达式语言

数据绑定库支持在布局文件中使用简单的表达式语言,这些表达式都包含在@{}内,并在编译时被处理和转换成Java代码。例如:

<TextView  android:text="@{viewModel.user.name + ', ' + viewModel.user.age + ' years old'}" />

3.5 性能优化

在处理大量数据时,合理的数据绑定可以减少内存占用并提高渲染速度。通过合理设计数据模型和布局,以及使用如RecyclerView等高效的UI组件,可以进一步提升应用的性能。

四、MVVM模式的优缺点分析

优点

  1. 高内聚低耦合:MVVM模式将UI逻辑与业务逻辑分离,提高了代码的可维护性和扩展性。
  2. 易于测试:ViewModel独立于View,使得业务逻辑可以更容易地进行单元测试。
  3. 数据驱动UI:通过数据绑定,UI可以自动响应数据的变化,减少了手动更新UI的工作量。

缺点

  1. 学习曲线:相对于传统的MVC模式,MVVM模式需要更多的时间来学习和理解。
  2. 复杂度增加:在小型项目中,MVVM模式可能会增加不必要的复杂度。

结论

MVVM模式是一种强大的架构模式,它通过将应用程序分为Model、View和ViewModel三个部分,实现了用户界面与业务逻辑的分离。在Android开发中,结合数据绑定技术,可以更有效地管理数据与UI之间的交互,减少代码冗余,提升应用性能。通过实践MVVM模式,开发者可以构建更清晰、可维护性更高的应用架构,提高开发效率和应用质量。

相关文章:

Android架构组件:MVVM模式的实战应用与数据绑定技巧

目录 引言 一、MVVM模式概述 1.1 MVVM模式简介 1.2 MVVM模式的优势 二、MVVM模式的实现 2.1 项目环境配置 2.2 创建MVVM组件 2.2.1 创建数据模型 2.2.2 创建数据仓库 2.2.3 创建ViewModel 2.2.4 创建布局文件 2.2.5 创建RecyclerView适配器 2.3 在Activity中绑定V…...

调用系统的录音设备提示:line with format PCM_SIGNED 16000.0 Hz

javax.sound.sampled.LineUnavailableException: line with format PCM_SIGNED 16000.0 Hz, 8 bit, mono, 1 bytes/frame, not supported. 打开 设置->隐私->麦克风->允许应用访问你的麦克风 与 16000Hz没关系 与 16000Hz没关系 与 16000Hz没关系...

android BLE 蓝牙的连接(二)

下面是基于实际的项目得到的具体步骤及核心代码 1、权限问题 先判断手机是否满足android4.3以上版本&#xff0c;再判断手机是否开启蓝牙 主要涉及蓝牙权限和位置权限&#xff0c;注意不同android版本之间权限申请的差异&#xff0c;以及android权限动态申请和静态申请的区别 …...

改编pikachu的打靶经历(题目不全)

前言 题目很少&#xff0c;只做了一些。正常版本的&#xff0c;完整的pikachu可参考下面这个师傅写的 https://www.cnblogs.com/henry666/p/16947270.html xss &#xff08;get&#xff09;反射xss 先尝试 1 这里有长度限制&#xff0c;而且&#xff0c;我改了长度&#xf…...

Linux进阶 修改文件所有者

修改文件所属组群——chgrp 修改文件所属组群很简单-chgrp命令,就是change group的缩写(我们可以利用这些来记忆命令) 语法:chgrp 组群 文件名/目录 举例: [root@redhat ~]# groupadd groupa[root@redhat ~]# groupadd groupb[root@redhat ~]# useradd -g groupa zgz[r…...

第312题|二重积分求旋转体体积(二)|武忠祥老师每日一题

解题思路&#xff1a;先画出图像&#xff0c;再利用旋转体体积计算公式进行解题。 1. 旋转体体积计算公式&#xff1a; 2.点到直线计算公式&#xff1a; 有了上面两条知识储备之后我们开始计算。 第一步&#xff1a;先计算出点到直线的距离&#xff1a; ymx&#xff0c;y-mx…...

redis基本数据结构-set

文章目录 1. set的基本介绍1.1. set底层结构之hash表的简单介绍1.2. 常用命令 2. 常见的业务场景2.1. 标签系统2.2. 社交网络好友关系 1. set的基本介绍 参考链接&#xff1a;https://mp.weixin.qq.com/s/srkd73bS2n3mjIADLVg72A redis 的 set 数据结构是一个无序的集合&#…...

Android 应用安装-提交阶段

经过前面准备、浏览、协调这些步骤&#xff0c;马上要进入提交阶段了。所谓提交&#xff0c;就是把这些安装应用的相关信息和状态都放到系统中。对于已安装普通应用&#xff0c;它其实分为两个步骤&#xff0c;先卸载旧包&#xff0c;再安装新包。当然&#xff0c;如果是新安装…...

强化学习Reinforcement Learning|Q-Learning|SARSA|DQN以及改进算法

一、强化学习RL 强化学习是机器学习的一个重要的分支&#xff0c;是一种有效的工具&#xff0c;在文献中被广泛用于解决MDP问题。在一个强化学习过程中&#xff0c;一个智能体只能通过和它所处的环境互动学习最优策略。特别地&#xff0c;智能体首先观察自己当前的状态&#xf…...

【HarmonyOS NEXT开发】如何设置水平/垂直方向的左/居中/右对齐——RelativeContainer的AlignRules设置

文章目录 【HarmonyOS NEXT开发】如何设置水平/垂直方向的左/居中/右对齐——RelativeContainer的AlignRules设置RelativeContainer 和 AlignRules 的关系AlignRules 语法详解 【HarmonyOS NEXT开发】如何设置水平/垂直方向的左/居中/右对齐——RelativeContainer的AlignRules设…...

java之认识异常

在 Java 中&#xff0c;异常&#xff08;Exception&#xff09;用于处理程序运行时出现的错误或异常情况。Java 的异常处理机制基于 try, catch, finally 和 throw 关键字。 1.异常的分类&#xff1a; 1.1&#xff1a;检查型异常(CheckedException)&#xff1a; 定义:程序在…...

JSON处理工具类

JSON处理工具类 import org.json.JSONArray; import org.json.JSONObject;import java.util.ArrayList; import java.util.List;/*** JSON处理工具类*/ public class JsonUtils {/****将json字符串转为map* param json* return java.util.Map<java.lang.String, java.lang.O…...

2022高教社杯全国大学生数学建模竞赛C题 问题一(2) Python代码演示

目录 1.2 结合玻璃的类型,分析文物样品表面有无风化化学成分含量的统计规律数据预处理绘图热力图相关系数图百分比条形图箱线图小提琴图直方图KED图描述性统计分析偏度系数峰度系数其它统计量1.2 结合玻璃的类型,分析文物样品表面有无风化化学成分含量的统计规律 数据预处理 …...

ARACom Proxy Class API 概念

1. Proxy Class 概述 生成方式&#xff1a;Proxy Class 是从 AutoSar 元模型的服务接口描述中生成的&#xff0c;ara::com 标准化了其接口&#xff0c;AP 产品供应商的工具链会生成实现该接口的代理实现类。 命名空间&#xff1a;ara::com 期望代理相关的工件在命名空间 “pro…...

【Scala入门学习】基本数据类型和变量声明

1. 基本数据类型 scala 的基本类型有 9种&#xff1a; Byte、Char、Short、Int、Long、Float、Double、Boolean、Unit Scala中没有基本数据类型的概念&#xff0c;所有的类型都是对象。 AnyVal&#xff1a;代表所有基本类型。 AnyRef&#xff1a;代表所以引用类型&#xff…...

C#基础(13)结构体

前言 随着函数的讲解完成&#xff0c;我想你已经初步有了写一些复杂逻辑功能的能力&#xff0c;但是我们会发现其实在我们大部分实际开发情况中&#xff0c;很多我们需要写的变量可能不只有一个属性。 他可能有很多变量&#xff0c;那这时候我们如果要把这些变量集中到一个东…...

Excel图片批量插入单元格排版处理插件【图片大师】

为了方便大家在图片的插入排版的重复工作中解放出来&#xff0c;最近发布了一款批量插入图片的插件&#xff0c;欢迎大家下载&#xff0c;免费试用。 这是图片的文件夹&#xff1a; 主要功能如下: 1&#xff0c;匹配单元格名称的多张图批量插入到一个单元格 该功能支持设置图…...

应用性能优化实践(二)提升应用启动和响应速度

一、提升应用启动和响应速度的方法 1、冷启动过程简介 应用启动时&#xff0c;后台无该应用的进程&#xff0c;需要创建新的进程&#xff0c;这种启动方式叫冷启动。 2、使用异步加载 使用异步加载可以在后台线程中处理耗时操作&#xff0c;从而提升应用响应速度。 3、延迟加载…...

接口测试系列文章专题

在你眼中什么是接口 HTTP协议 什么是接口测试 接口测试之工具 fiddler工具的原理 fiddler工具界面详解 fiddler工具的基本使用 fiddler使如何对手机app进行抓包的呢 fiddler手机app抓包教程 Charles自定义接口返回的数据内容 常用接口工具postman的基本使用方式 pos…...

Unity Hub自动安装指定版本Unity的Android开发环境

Unity开发Android环境要求SDK、DNK、JDK、Gradle版本都要对才能发布APK&#xff0c;自己去配置很容易出错。Unity Hub可以自动安装指定版本Unity的Android开发环境。 1.安装国内用的UnityHub&#xff08;我这里用的3.3.2-c6&#xff09; 2.找到对应的Unity版本 3.点击【从Unit…...

MPNet:旋转机械轻量化故障诊断模型详解python代码复现

目录 一、问题背景与挑战 二、MPNet核心架构 2.1 多分支特征融合模块(MBFM) 2.2 残差注意力金字塔模块(RAPM) 2.2.1 空间金字塔注意力(SPA) 2.2.2 金字塔残差块(PRBlock) 2.3 分类器设计 三、关键技术突破 3.1 多尺度特征融合 3.2 轻量化设计策略 3.3 抗噪声…...

HTML 语义化

目录 HTML 语义化HTML5 新特性HTML 语义化的好处语义化标签的使用场景最佳实践 HTML 语义化 HTML5 新特性 标准答案&#xff1a; 语义化标签&#xff1a; <header>&#xff1a;页头<nav>&#xff1a;导航<main>&#xff1a;主要内容<article>&#x…...

微信小程序之bind和catch

这两个呢&#xff0c;都是绑定事件用的&#xff0c;具体使用有些小区别。 官方文档&#xff1a; 事件冒泡处理不同 bind&#xff1a;绑定的事件会向上冒泡&#xff0c;即触发当前组件的事件后&#xff0c;还会继续触发父组件的相同事件。例如&#xff0c;有一个子视图绑定了b…...

基于距离变化能量开销动态调整的WSN低功耗拓扑控制开销算法matlab仿真

目录 1.程序功能描述 2.测试软件版本以及运行结果展示 3.核心程序 4.算法仿真参数 5.算法理论概述 6.参考文献 7.完整程序 1.程序功能描述 通过动态调整节点通信的能量开销&#xff0c;平衡网络负载&#xff0c;延长WSN生命周期。具体通过建立基于距离的能量消耗模型&am…...

视频字幕质量评估的大规模细粒度基准

大家读完觉得有帮助记得关注和点赞&#xff01;&#xff01;&#xff01; 摘要 视频字幕在文本到视频生成任务中起着至关重要的作用&#xff0c;因为它们的质量直接影响所生成视频的语义连贯性和视觉保真度。尽管大型视觉-语言模型&#xff08;VLMs&#xff09;在字幕生成方面…...

SpringBoot+uniapp 的 Champion 俱乐部微信小程序设计与实现,论文初版实现

摘要 本论文旨在设计并实现基于 SpringBoot 和 uniapp 的 Champion 俱乐部微信小程序&#xff0c;以满足俱乐部线上活动推广、会员管理、社交互动等需求。通过 SpringBoot 搭建后端服务&#xff0c;提供稳定高效的数据处理与业务逻辑支持&#xff1b;利用 uniapp 实现跨平台前…...

反射获取方法和属性

Java反射获取方法 在Java中&#xff0c;反射&#xff08;Reflection&#xff09;是一种强大的机制&#xff0c;允许程序在运行时访问和操作类的内部属性和方法。通过反射&#xff0c;可以动态地创建对象、调用方法、改变属性值&#xff0c;这在很多Java框架中如Spring和Hiberna…...

PL0语法,分析器实现!

简介 PL/0 是一种简单的编程语言,通常用于教学编译原理。它的语法结构清晰,功能包括常量定义、变量声明、过程(子程序)定义以及基本的控制结构(如条件语句和循环语句)。 PL/0 语法规范 PL/0 是一种教学用的小型编程语言,由 Niklaus Wirth 设计,用于展示编译原理的核…...

为什么要创建 Vue 实例

核心原因:Vue 需要一个「控制中心」来驱动整个应用 你可以把 Vue 实例想象成你应用的**「大脑」或「引擎」。它负责协调模板、数据、逻辑和行为,将它们变成一个活的、可交互的应用**。没有这个实例,你的代码只是一堆静态的 HTML、JavaScript 变量和函数,无法「活」起来。 …...

tomcat入门

1 tomcat 是什么 apache开发的web服务器可以为java web程序提供运行环境tomcat是一款高效&#xff0c;稳定&#xff0c;易于使用的web服务器tomcathttp服务器Servlet服务器 2 tomcat 目录介绍 -bin #存放tomcat的脚本 -conf #存放tomcat的配置文件 ---catalina.policy #to…...