WPF運行時替換方法實現mvvm自動觸發(fā)刷新
前言
我們知道,使用wpf的綁定功能,代碼觸發(fā)界面改變時需要在屬性的setter觸發(fā)PropertyChanged事件,以達到界面刷新的效果。上一章我們簡化了觸發(fā)流程,但是依然需要在每個屬性的setter中調用方法。本章將再進一步簡化,實現setter不需要調方法就可以自動觸發(fā)界面刷新。
一、如何實現
1、反射獲取屬性
通過反射獲取類的所有公有屬性
var propertyInfos = type.GetProperties(bindingAttr: BindingFlags.Instance | BindingFlags.Public);
2、定義替換方法
定義的用于替換屬性setter的方法,確保參數類型兼容。設置NoInlining確保不會被內聯(lián)優(yōu)化失去函數地址。再方法中觸發(fā)RaisePropertyChangedEvent。
[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的實現。
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、接口
/// <summary>
/// ViewModelBase,繼承此類可以簡化屬性的定義,不需要手動觸發(fā)RaisePropertyChangedEvent。
/// 用法:繼承此類,屬性為公有,set為公有且非內聯(lián),設置屬性就會自動觸發(fā)mvvm的binding。
/// 實驗性質,其他.net版本無效,在.net6.0是穩(wěn)定的,。x64、x86,debug和release都可以使用。release需要給set設置[MethodImpl(MethodImplOptions.NoInlining)],否則無法實現函數交換。
/// 目前支持64個屬性,單個屬性(struct)最大128字節(jié),需要更多可以自行調用GenSetters生成代碼。
/// </summary>
public abstract class ViewModelBase : INotifyPropertyChanged
{
public event PropertyChangedEventHandler PropertyChanged;
public ViewModelBase();
//依然提供此方法用于手動觸發(fā)
protected void RaisePropertyChangedEvent([System.Runtime.CompilerServices.CallerMemberName] string propertyName = "");
}
2、項目
vs2022 .net 6.0項目。
注:目前版本只能在.net6.0中正常使用,x64、x86、debug、release都沒問題。其他.net版本大概率無效果或者異常。
三、使用示例
倒計時
(1)繼承ViewModelBase
public class TimeTick : ViewModelBase
(2)定義屬性
set為公有,以及[MethodImpl(MethodImplOptions.NoInlining)]避免內聯(lián)。支持非缺省set方法體,即可以在set中加入一些邏輯。
public class TimeTick : ViewModelBase
{
public double Seconds { get; [MethodImpl(MethodImplOptions.NoInlining)] set; }=60;
}
(3)屬性賦值
public TimeTick()
{
var time = new DispatcherTimer() { Interval = TimeSpan.FromMilliseconds(1000) };
time.Tick += (s, e) =>Seconds--;
time.Start();
}
(4)窗口關聯(lián)ViewModel
public partial class MainWindow : Window
{
public MainWindow()
{
InitializeComponent();
DataContext = new TimeTick();
}
}
(5)xaml綁定
<TextBlock Text="{Binding Seconds}" />
效果預覽

總結
替換函數原理很簡單,但是具體實現還是比較麻煩的,尤其是需要適配不同的.net版本,本文目前只支持.net6.0。還有就是函數的參數,引用和值類型的區(qū)分,以及值類型的傳值兼容,這些都是通過多次嘗試才找個合理的方式。通過本文簡化的ViewModelBase使用變的非常方便了,除了需要給set添加非內聯(lián)屬性,其他已經和普通屬性沒有區(qū)別。
到此這篇關于WPF運行時替換方法實現mvvm自動觸發(fā)刷新的文章就介紹到這了,更多相關WPF mvvm自動觸發(fā)刷新內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!

