理解并实现 Android MVVM 架构

Android MVVM 架构

在现代 Android 开发中,维护一个清晰、可测试和模块化的代码库变得至关重要。Google 推荐的架构实践是使用 MVVM(Model-View-ViewModel)模式,它是响应式编程思想在 Android 应用开发中的体现。在本文中,我们将深入探讨 MVVM 架构,并通过一个简单的示例演示如何在 Android 应用程序中实现它。

MVVM 架构概述

MVVM 架构分为三个核心组件:

  • Model - 表示应用程序的数据和业务逻辑,例如网络请求、数据库交互等。
  • View - 表示应用程序的 UI 组件,例如 Activities 和 Fragments。
  • ViewModel - 充当 View 和 Model 之间的中介。它处理从 Model 获取的数据,使其适用于 View 显示,同时也响应 View 的用户交互。

以下是每个组件的职责:

Model

Model 包含了应用程序的数据处理部分。通常,Model 可以进一步分为几个部分:

  • Repository: 提供数据的 API,它决定从本地数据库或网络获取数据。
  • Local Data Source: 如 SQLite 数据库或 Room。
  • Remote Data Source: 如 Retrofit 或其他网络请求库。

View

View 是用户看到并与之交互的屏幕。在 Android 中,它通常指的是 ActivityFragment。View 只负责展示数据,并将用户操作通知给 ViewModel。它应该尽可能地"愚蠢",不包含任何业务逻辑。

ViewModel

ViewModel 是 UI 的数据提供者。它不直接请求数据,而是通过 Repository 来管理数据。ViewModel 也不直接引用 View,它通过 LiveData 或其他观察者模式来通知 UI 数据的变化。

实现 MVVM 架构

让我们通过构建一个简单的用户列表界面来理解 MVVM 的实现。我们将使用以下 Jetpack 组件:

  • LiveData
  • ViewModel
  • View Binding

步骤 1: 配置依赖

首先,在你的 build.gradle (Module: app) 文件中添加以下依赖:

dependencies {
    // ViewModel 和 LiveData
    implementation "androidx.lifecycle:lifecycle-viewmodel-ktx:2.3.1"
    implementation "androidx.lifecycle:lifecycle-livedata-ktx:2.3.1"
    
    // View Binding
    buildFeatures {
        viewBinding true
    }
}

步骤 2: 创建 Model

// User.java
public class User {
    private int id;
    private String name;

    public User(int id, String name) {
        this.id = id;
        this.name = name;
    }

    // Getters and setters
    public int getId() {
        return id;
    }

    public String getName() {
        return name;
    }
}

步骤 3: 创建 ViewModel

// MainViewModel.java
import androidx.lifecycle.MutableLiveData;
import androidx.lifecycle.ViewModel;

import java.util.Arrays;
import java.util.List;

public class MainViewModel extends ViewModel {
    private MutableLiveData<List<User>> users = new MutableLiveData<>();

    public MainViewModel() {
        loadUsers();
    }

    public MutableLiveData<List<User>> getUsers() {
        return users;
    }

    private void loadUsers() {
        // 通常这里会是调用 Repository 层的代码
        List<User> dummyUsers = Arrays.asList(new User(1, "Alice"), new User(2, "Bob"));
        users.setValue(dummyUsers);
    }
}

步骤 4: 创建 View

activity_main.xml:

<!-- activity_main.xml -->
<LinearLayout ...>
    <TextView
        android:id="@+id/textView"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content" />
</LinearLayout>

MainActivity.kt 中使用 View Binding 和 ViewModel:

// MainActivity.kt
package com.example.myapp;

import android.os.Bundle;
import androidx.appcompat.app.AppCompatActivity;
import androidx.lifecycle.ViewModelProvider;
import com.example.myapp.databinding.ActivityMainBinding;

public class MainActivity extends AppCompatActivity {

    private MainViewModel mainViewModel;
    private ActivityMainBinding binding;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);

        // 初始化 View Binding
        binding = ActivityMainBinding.inflate(getLayoutInflater());
        setContentView(binding.getRoot());

        // 初始化 ViewModel
        mainViewModel = new ViewModelProvider(this).get(MainViewModel.class);

        // 观察 ViewModel 中的 LiveData 对象
        mainViewModel.getUsers().observe(this, users -> {
            // 更新 UI
            StringBuilder userInfo = new StringBuilder();
            for (User user : users) {
                userInfo.append(user.getName()).append("\n");
            }
            binding.textView.setText(userInfo.toString());
        });
    }
}

记住,LiveData 的观察模式确保了只有在 ActivityFragment 处于活动状态时,才会更新 UI 组件。

步骤 5: 测试你的应用

现在你可以运行你的应用程序,并看到 TextView 中显示的用户列表。这个示例非常基础,但它展示了如何将数据从 ViewModel 传递到 View 而不需要 View 知道数据来源的复杂性。

结论

MVVM 是一种强大且灵活的架构模式,它促进了代码的分离和模块化。通过利用 LiveData 和 ViewModel,我们可以创建响应式应用程序,这些应用程序能够优雅地处理生命周期事件和数据管理。此外,View Binding 的使用进一步简化了 UI 代码,使得我们能够避免 findViewById 的繁琐并减少出错的几率。

在实际开发中,你可能还需要引入如 Data Binding、Room、Navigation 等其他 Jetpack 组件来进一步提升应用架构的效率和功能性。

希望这篇博客能帮助你理解并开始使用 MVVM 架构来构建你的 Android 应用。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,470评论 6 501
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,393评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,577评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,176评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,189评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,155评论 1 299
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,041评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,903评论 0 274
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,319评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,539评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,703评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,417评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,013评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,664评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,818评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,711评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,601评论 2 353

推荐阅读更多精彩内容