WPF命令绑定核心是ICommand接口,推荐用RelayCommand实现解耦与自动启停;需在属性变更后调用CommandManager.InvalidateRequerySuggested刷新状态,RoutedCommand适用于跨控件共享或快捷键场景。
WPF 中命令绑定的核心是 ICommand 接口,它让 UI 操作(比如按钮点击)和业务逻辑解耦,比直接写 Click 事件更灵活、可测试、支持自动启停(如按钮灰化)。下面用最常用也最实用的方式讲清楚怎么用。
WPF 自身没提供 ICommand 的默认实现,但社区广泛使用 RelayCommand(也叫 DelegateCommand)——它用 Action 和 Func 封装执行逻辑和判断逻辑,轻量又直观。
public class RelayCommand : ICommand
{
private readonly Action _execute;
private readonly Func_canExecute;
public RelayCommand(Action execute, FunccanExecute = null)
{
_execute = execute;
_canExecute = canExecute?? (() => true);
}
public bool CanExecute(object parameter) => _canExecute();
public void Execute(object parameter) => _execute();
public event EventHandler CanExecuteChanged
{
add => CommandManager.RequerySuggested += value;
remove => CommandManager.RequerySuggested -= value;
}
}
public class MainViewModel
{
public ICommand SaveCommand { get; }
public MainViewModel()
{
SaveCommand = new RelayCommand(OnSave, CanSave);
}
private void OnSave() => MessageBox.Show("已保存");
private bool CanSave() => !string.IsNullOrWhiteSpace(Title); // 假设有个 Title 属性
}
把按钮的 Command 属性直接绑定到 ViewModel 的 ICommand 属性,WPF 会自动调用 Execute,并根据 CanExecute 结果控制是否启用。
命令能否执行不是一成不变的,比如输入框为空时“保存”按钮应禁用。WPF 不会自动监听你的属性变化,得手动通知。
private string _title;
public string Title
{
get => _title;
set
{
_title = value;
OnPropertyChanged(); // INotifyPropertyChanged 实现
CommandManager.InvalidateRequerySuggested(); // 关键!告诉 WPF 重新查 CanExecute
}
}
如果你需要响应 Ctrl+S、或让多个控件共用一个命令(比如“复制”在菜单和工具栏都可用),用 WPF 自带的 RoutedCommand 更合适。
public static class ApplicationCommands
{
public static readonly RoutedCommand Save = new RoutedCommand();
}
Executed="OnSaveExecuted"
CanExecute="OnSaveCanExecute"/>
private void OnSaveExecuted(object sender, ExecutedRoutedEventArgs e)
{
MessageBox.Show("RoutedCommand 保存");
}
private void OnSaveCanExecute(object sender, CanExecuteRoutedEventArgs e)
{
e.CanExecute = !string.IsNullOrEmpty(Title);
}
基本上就这些。RelayCommand + Binding 是日常开发主力,RoutedCommand 用在需要路由或跨控件共享的场景。不复杂但容易忽略 CommandManager.InvalidateRequerySuggested 这一步——少了它,按钮状态就不会自动更新。