0

0

Avalonia怎么在单元测试中测试ViewModel Avalonia单元测试教程

幻夢星雲

幻夢星雲

发布时间:2026-01-11 10:33:09

|

544人浏览过

|

来源于php中文网

原创

在 Avalonia 中测试 ViewModel 的核心是将其视为普通 C# 类,不依赖 UI 组件,通过验证属性变更通知、命令执行及状态变化来确保逻辑正确,依赖服务需接口注入并用 Moq 等模拟。

avalonia怎么在单元测试中测试viewmodel avalonia单元测试教程

在 Avalonia 中对 ViewModel 做单元测试,核心思路是:把 ViewModel 当作普通 C# 类来测,不依赖 UI、不启动窗口、不触发渲染。只要它实现了 INotifyPropertyChanged、用了命令(ICommand)、有业务逻辑或状态流转,就能独立验证。

ViewModel 必须可测试的前提

确保你的 ViewModel 基类已正确支持属性变更通知和命令机制:

  • 继承自 ReactiveUI.ViewModelCommunityToolkit.Mvvm.ObservableObject,或手动实现 INotifyPropertyChanged
  • 使用 RelayCommand / ReactiveCommand / AsyncRelayCommand 定义命令,而非直接写事件处理方法
  • 避免在 ViewModel 中调用 Application.CurrentDispatcher.UIThreadDialogHost 等 Avalonia UI 特定对象
  • 依赖的服务(如数据访问、网络请求)应通过接口注入,方便在测试中用 Moq 或 Fake 替换

写一个基础单元测试(以 CommunityToolkit.Mvvm 为例)

假设你有如下 ViewModel:

public partial class LoginViewModel : ObservableObject
{
    [ObservableProperty]
    private string _account;
[ObservableProperty]
private string _pwd;

[ICommand]
public void Login()
{
    if (!string.IsNullOrWhiteSpace(Account) && Account.Length > 2)
        IsLoggedIn = true;
}

[ObservableProperty]
private bool _isLoggedIn;

}

ClippingMagic
ClippingMagic

魔术般地去除图片背景

下载

对应测试代码(用 xUnit + Moq):

  • 新建 xUnit 测试项目,引用被测项目和 CommunityToolkit.Mvvm
  • 创建测试类,实例化 ViewModel
  • 验证属性赋值后是否触发通知
  • 调用命令后检查状态变化

示例:

[Fact]
public void Login_WhenAccountValid_SetsIsLoggedInToTrue()
{
    var vm = new LoginViewModel();
    vm.Account = "admin";
    vm.Pwd = "123";
vm.Login();

Assert.True(vm.IsLoggedIn);

}

测试属性变更通知(INPC)

关键是要验证 PropertyChanged 事件是否被正确触发:

  • 订阅 PropertyChanged 事件,记录触发的属性名
  • 修改属性,检查是否收到对应事件
  • 推荐用 CommunityToolkit.MvvmSetProperty[ObservableProperty],它们默认保障通知逻辑正确

简单验证方式:

var vm = new LoginViewModel();
string? raisedProp = null;
vm.PropertyChanged += (_, e) => raisedProp = e.PropertyName;

vm.Account = "test"; Assert.Equal("Account", raisedProp);

测试异步命令和响应式逻辑

如果用了 ReactiveUI,重点测 ReactiveCommand 的执行、CanExecute 变化、执行结果:

  • TestScheduler 控制时间流(适合复杂响应式链)
  • Execute 调用后,断言输出属性、服务调用次数、异常等
  • 模拟依赖服务返回 Task 或 Observable,例如用 Observable.Return(...)Task.FromResult(...)

例如测试登录失败场景:

var mockAuthService = new Mock();
mockAuthService.Setup(x => x.LoginAsync(It.IsAny(), It.IsAny()))
               .ReturnsAsync(false);

var vm = new LoginViewModel(mockAuthService.Object); vm.Account = "bad"; vm.Pwd = "pass"; await vm.LoginCommand.ExecuteAsync(null);

Assert.False(vm.IsLoggedIn);

基本上就这些。不复杂但容易忽略的是:别在 ViewModel 里做 UI 导航、弹窗、资源加载——那些该交给 View 或专门的导航服务,否则测试会卡住或失败。

相关专题

更多
硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1010

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

59

2025.10.17

php8.4实现接口限流的教程
php8.4实现接口限流的教程

PHP8.4本身不内置限流功能,需借助Redis(令牌桶)或Swoole(漏桶)实现;文件锁因I/O瓶颈、无跨机共享、秒级精度等缺陷不适用高并发场景。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

362

2025.12.29

c++主流开发框架汇总
c++主流开发框架汇总

本专题整合了c++开发框架推荐,阅读专题下面的文章了解更多详细内容。

78

2026.01.09

c++框架学习教程汇总
c++框架学习教程汇总

本专题整合了c++框架学习教程汇总,阅读专题下面的文章了解更多详细内容。

46

2026.01.09

学python好用的网站推荐
学python好用的网站推荐

本专题整合了python学习教程汇总,阅读专题下面的文章了解更多详细内容。

121

2026.01.09

学python网站汇总
学python网站汇总

本专题整合了学python网站汇总,阅读专题下面的文章了解更多详细内容。

11

2026.01.09

python学习网站
python学习网站

本专题整合了python学习相关推荐汇总,阅读专题下面的文章了解更多详细内容。

15

2026.01.09

俄罗斯手机浏览器地址汇总
俄罗斯手机浏览器地址汇总

汇总俄罗斯Yandex手机浏览器官方网址入口,涵盖国际版与俄语版,适配移动端访问,一键直达搜索、地图、新闻等核心服务。

71

2026.01.09

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
React 教程
React 教程

共58课时 | 3.5万人学习

国外Web开发全栈课程全集
国外Web开发全栈课程全集

共12课时 | 1.0万人学习

React核心原理新老生命周期精讲
React核心原理新老生命周期精讲

共12课时 | 1万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号