C# wpf 运行时替换方法实现mvvm自动触发刷新

文章目录


前言

我们知道,使用wpf的绑定功能,代码触发界面改变时需要在属性的setter触发PropertyChanged事件,以达到界面刷新的效果。上一章我们简化了触发流程,但是依然需要在每个属性的setter中调用方法。本章将再进一步简化,实现setter不需要调方法就可以自动触发界面刷新


一、如何实现?

1、反射获取属性

通过反射获取类的所有公有属性

csharp 复制代码
var propertyInfos = type.GetProperties(bindingAttr: BindingFlags.Instance | BindingFlags.Public);

2、定义替换方法

定义的用于替换属性setter的方法,确保参数类型兼容。设置NoInlining确保不会被内联优化失去函数地址。再方法中触发RaisePropertyChangedEvent。

csharp 复制代码
 [MethodImpl(MethodImplOptions.NoInlining)] 
 void Setter0_obj(object value) { 
  //此时Setter0_obj已经被替换成了属性的setter,调用会进入属性的setter中。
  Setter0_obj(value);
  RaisePropertyChangedEvent(_propertyInfos![0].Name); }

3、交换属性的setter方法

将定义的替换方法与属性的setter交换。MethodHooker.SwapMethod可以去搜索C#运行时替换函数的方法,本章的只是去掉了unsafe的实现。

csharp 复制代码
var oldSetter = propertyInfos[i].GetSetMethod();
if (oldSetter != null && oldSetter.IsPublic)
//定义了set且set为公有时才交换。
{
    MethodInfo newSetter= type.BaseType.GetMethod("Setter0_obj", BindingFlags.Instance | BindingFlags.NonPublic)!;
    MethodHooker.SwapMethod(oldSetter, newSetter);
}

二、完整代码

1、接口

csharp 复制代码
/// <summary>
/// ViewModelBase,继承此类可以简化属性的定义,不需要手动触发RaisePropertyChangedEvent。
/// 用法:继承此类,属性为公有,set为公有且非内联,设置属性就会自动触发mvvm的binding。
/// 实验性质,其他.net版本无效,在.net6.0是稳定的,。x64、x86,debug和release都可以使用。release需要给set设置[MethodImpl(MethodImplOptions.NoInlining)],否则无法实现函数交换。
/// 目前支持64个属性,单个属性(struct)最大128字节,需要更多可以自行调用GenSetters生成代码。
/// </summary>
public abstract class ViewModelBase : INotifyPropertyChanged
{
    public event PropertyChangedEventHandler PropertyChanged;
    public ViewModelBase();
    //依然提供此方法用于手动触发
    protected void RaisePropertyChangedEvent([System.Runtime.CompilerServices.CallerMemberName] string propertyName = "");
}

2、项目

vs2022 .net 6.0项目。
https://download.csdn.net/download/u013113678/89241921
注:目前版本只能在.net6.0中正常使用,x64、x86、debug、release都没问题。其他.net版本大概率无效果或者异常。属于实验性质,请根据需要下载。


三、使用示例

1、倒计时

(1)、继承ViewModelBase

csharp 复制代码
public class TimeTick : ViewModelBase

(2)、定义属性

set为公有 ,以及[MethodImpl(MethodImplOptions.NoInlining)]避免内联。支持非缺省set方法体,即可以在set中加入一些逻辑。

csharp 复制代码
public class TimeTick : ViewModelBase
{
    public double Seconds { get; [MethodImpl(MethodImplOptions.NoInlining)] set; }=60;
}

(3)、属性赋值

csharp 复制代码
public TimeTick()
{
    var time = new DispatcherTimer() { Interval = TimeSpan.FromMilliseconds(1000) };
    time.Tick += (s, e) =>Seconds--;     
    time.Start();
}

(4)、窗口关联ViewModel

csharp 复制代码
public partial class MainWindow : Window
{
    public MainWindow()
    {
        InitializeComponent();
        DataContext = new TimeTick();
    }
}

(5)、xaml绑定

xml 复制代码
 <TextBlock Text="{Binding Seconds}" />

效果预览


总结

以上就是今天讲述的内容,替换函数原理很简单,但是具体实现还是比较麻烦的,尤其是需要适配不同的.net版本,本文目前只支持.net6.0。还有就是函数的参数,引用和值类型的区分,以及值类型的传值兼容,这些都是通过多次尝试才找个合理的方式。通过本文简化的ViewModelBase使用变的非常方便了,除了需要给set添加非内联属性,其他已经和普通属性没有区别。

相关推荐
新手unity自用笔记14 小时前
项目-坦克大战学习-子弹的移动与销毁
笔记·学习·c#
qinzechen14 小时前
分享几个做题网站------学习网------工具网;
java·c语言·c++·python·c#
yufei-coder18 小时前
C# Windows 窗体开发基础
vscode·microsoft·c#·visual studio
dangoxiba18 小时前
[Unity Demo]从零开始制作空洞骑士Hollow Knight第十三集:制作小骑士的接触地刺复活机制以及完善地图的可交互对象
游戏·unity·visualstudio·c#·游戏引擎
AitTech18 小时前
深入理解C#中的TimeSpan结构体:创建、访问、计算与格式化
开发语言·数据库·c#
hiyo5851 天前
C#中虚函数和抽象函数的概念
开发语言·c#
芝麻科技1 天前
使用ValueConverters扩展实现枚举控制页面的显示
wpf·prism
开心工作室_kaic1 天前
基于微信小程序的校园失物招领系统的设计与实现(论文+源码)_kaic
c语言·javascript·数据库·vue.js·c#·旅游·actionscript
时光追逐者1 天前
WaterCloud:一套基于.NET 8.0 + LayUI的快速开发框架,完全开源免费!
前端·microsoft·开源·c#·.net·layui·.netcore
friklogff1 天前
【C#生态园】打造现代化跨平台应用:深度解析.NET桌面应用工具
开发语言·c#·.net