【大前端】【Android】把 Activity 重构成 MVVM 的对比示例

同一个功能:加载用户并显示名字

功能不变:

点按钮 → 请求用户 → 显示用户名 → 失败提示


一、❌ 传统 Activity 写法(所有东西都在 Activity)

Activity(UI类)

复制代码
public class UserActivity extends AppCompatActivity {

    TextView tvName;
    Button btnLoad;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_user);

        tvName = findViewById(R.id.tvName);
        btnLoad = findViewById(R.id.btnLoad);

        btnLoad.setOnClickListener(v -> {
            // ① UI 事件
            loadUser();
        });
    }

    private void loadUser() {
        // ② 业务逻辑
        showLoading();

        Api.getUser(new Callback<User>() {
            @Override
            public void onSuccess(User user) {
                // ③ 数据处理
                tvName.setText(user.getName());   // 改 UI
                hideLoading();
            }

            @Override
            public void onError(String msg) {
                // ④ 错误处理
                Toast.makeText(UserActivity.this, msg, Toast.LENGTH_SHORT).show();
                hideLoading();
            }
        });
    }
}

❗问题在哪?

问题 原因
Activity 很胖 所有职责都在
不可测试 依赖 Android
UI & 逻辑耦合 直接操作 View
状态不可控 loading / error 到处写

二、✅ 重构目标(先不看代码)

我们要做到:

复制代码
Activity:我只管展示
ViewModel:我管逻辑和状态

Activity 不再知道

  • 请求怎么发

  • 什么时候成功 / 失败

  • 状态怎么切换


三、✅ MVVM 写法(拆开)


1️⃣ ViewModel:专门"管状态"

复制代码
public class UserViewModel extends ViewModel {

    // 用户名状态
    private MutableLiveData<String> userName = new MutableLiveData<>();

    // loading 状态
    private MutableLiveData<Boolean> loading = new MutableLiveData<>();

    // 错误信息
    private MutableLiveData<String> errorMsg = new MutableLiveData<>();

    public LiveData<String> getUserName() {
        return userName;
    }

    public LiveData<Boolean> getLoading() {
        return loading;
    }

    public LiveData<String> getErrorMsg() {
        return errorMsg;
    }

    public void loadUser() {
        loading.setValue(true);

        Api.getUser(new Callback<User>() {
            @Override
            public void onSuccess(User user) {
                userName.setValue(user.getName()); // 只改数据
                loading.setValue(false);
            }

            @Override
            public void onError(String msg) {
                errorMsg.setValue(msg);             // 只抛状态
                loading.setValue(false);
            }
        });
    }
}

📌 注意一个关键点

ViewModel 从头到尾没有一句 UI 代码


2️⃣ Activity:变成"数据监听器"

复制代码
public class UserActivity extends AppCompatActivity {

    private UserViewModel viewModel;
    private TextView tvName;
    private ProgressBar progressBar;
    private Button btnLoad;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_user);

        tvName = findViewById(R.id.tvName);
        progressBar = findViewById(R.id.progressBar);
        btnLoad = findViewById(R.id.btnLoad);

        viewModel = new ViewModelProvider(this).get(UserViewModel.class);

        // ① 监听数据
        viewModel.getUserName().observe(this, name -> {
            tvName.setText(name);
        });

        viewModel.getLoading().observe(this, show -> {
            progressBar.setVisibility(show ? View.VISIBLE : View.GONE);
        });

        viewModel.getErrorMsg().observe(this, msg -> {
            if (msg != null) {
                Toast.makeText(this, msg, Toast.LENGTH_SHORT).show();
            }
        });

        // ② 把点击事件"交出去"
        btnLoad.setOnClickListener(v -> {
            viewModel.loadUser();
        });
    }
}

四、最重要的一行对比(🔥核心)

以前

复制代码
tvName.setText(user.getName());

现在

复制代码
userName.setValue(user.getName());

👉 UI 更新从"命令式"变成"数据驱动"


五、用一句话理解"重构发生了什么"

Activity 不再"做事"
ViewModel 不再"碰 UI"
它们通过"状态"交流


六、责任变化对照表(一定要看)

职责 以前 现在
点击处理 Activity Activity(转交)
网络请求 Activity ViewModel
成功失败判断 Activity ViewModel
UI 更新 Activity Activity
状态来源 Activity ViewModel

七、已经"半只脚进 MVVM 了"的标志

如果现在能理解这句话:

Activity 不知道"什么时候该改 UI",
它只知道"数据变了我就展示"

那已经抓住 MVVM 的核心原理 了。

相关推荐
qq_1702647531 分钟前
unity出安卓年龄分级的arr包问题
android·unity·游戏引擎
kejiashao2 小时前
Android View的绘制流程及事件分发机制
android
小蜜蜂嗡嗡3 小时前
flutter实现付费解锁内容的遮挡
android·flutter
进击的cc3 小时前
拒绝背诵!一文带你打穿 Android ANR 发生的底层全链路
android·面试
进击的cc3 小时前
App 启动优化全家桶:别再只盯着 Application 了,热启动优化你真的做对了吗?
android·面试
彭波3963 小时前
安卓手机端安装xapk、apkm软件!怎样安装xapk软件?安卓的apk和XAPK的区别?附教程
android·智能手机
Yang-Never4 小时前
ADB ->adb shell perfetto 抓取 trace 指令
android·开发语言·adb·android studio
2501_937189237 小时前
莫凡电视:地方台专属聚合 稳定直播播放工具
android·源码·源代码管理
耶叶8 小时前
Android 新权限申请模型(Activity Result API)
android
阿拉斯攀登8 小时前
【RK3576 安卓 JNI/NDK 系列 04】JNI 核心语法(下):字符串、数组与对象操作
android·驱动开发·rk3568·瑞芯微·rk安卓驱动·jni字符串操作