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;

}

Magician
Magician

Figma插件,AI生成图标、图片和UX文案

下载

对应测试代码(用 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接口等等。

1017

2023.10.19

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

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

62

2025.10.17

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

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

400

2025.12.29

Java 桌面应用开发(JavaFX 实战)
Java 桌面应用开发(JavaFX 实战)

本专题系统讲解 Java 在桌面应用开发领域的实战应用,重点围绕 JavaFX 框架,涵盖界面布局、控件使用、事件处理、FXML、样式美化(CSS)、多线程与UI响应优化,以及桌面应用的打包与发布。通过完整示例项目,帮助学习者掌握 使用 Java 构建现代化、跨平台桌面应用程序的核心能力。

36

2026.01.14

php与html混编教程大全
php与html混编教程大全

本专题整合了php和html混编相关教程,阅读专题下面的文章了解更多详细内容。

18

2026.01.13

PHP 高性能
PHP 高性能

本专题整合了PHP高性能相关教程大全,阅读专题下面的文章了解更多详细内容。

34

2026.01.13

MySQL数据库报错常见问题及解决方法大全
MySQL数据库报错常见问题及解决方法大全

本专题整合了MySQL数据库报错常见问题及解决方法,阅读专题下面的文章了解更多详细内容。

19

2026.01.13

PHP 文件上传
PHP 文件上传

本专题整合了PHP实现文件上传相关教程,阅读专题下面的文章了解更多详细内容。

16

2026.01.13

PHP缓存策略教程大全
PHP缓存策略教程大全

本专题整合了PHP缓存相关教程,阅读专题下面的文章了解更多详细内容。

6

2026.01.13

热门下载

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

精品课程

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

共58课时 | 3.6万人学习

国外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号