WPF跨线程调用和Dispatcher

本文介绍在WPF应用程序中如何正确地进行跨线程更新UI的操作。为了避免线程访问冲突导致的错误,文中提供了使用Dispatcher进行UI更新的具体实例,并提及了进一步理解WPF线程模型的重要性。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

在winform中默认不允许跨线程操作控件,但可通过设置控件属性CheckForIllegalCrossThreadCalls = false。在WPF中同样也是禁止跨线程操作控件的,会提示“调用线程无法访问此对象,因为另一个线程拥有该对象。”的错误。

这里写图片描述
正确的操作是使用Dispatcher。

        void Test()
        {
            int i = 0;
            while (i < 100)
            {
                //Diapatcher + Action + lambda 匿名函数
                this.lblValue.Dispatcher.Invoke(new Action(() => this.lblValue.Content = i.ToString()));

                Thread.Sleep(10);
                i++;
            }
        }

        //点击按钮开始计时
        //刷新lblValue控件的值
        private async void btnRun_Click(object sender, RoutedEventArgs e)
        {
            //async/await + Task + Action +lambda 匿名函数
            await Task.Run(new Action(() => Test()));
        }

继续深入Dispatcher就需要了解WPF线程模型,暂时还没搞懂,可以参考一站式WPF–线程模型和Dispatcher

WPF中,由于UI线程负责更新用户界面,因此直接在非UI线程中修改控件的属性可能会导致应用程序阻塞或引发未授权操作错误。为了安全地跨线程更新UI,通常需要借助以下几个步骤: 1. **使用Dispatcher**:在非UI线程上,你可以创建一个`DispatcherOperation`或者`Dispatcher.BeginInvoke`来异步将任务提交给UI线程执行。例如: ```csharp private void UpdateControlOnUIThread() { Dispatcher.Invoke(() => { YourControl.Value = newValue; }); } ``` 或者 ```csharp Action action = () => YourControl.Value = newValue; Dispatcher.BeginInvoke(action); ``` 2. **事件委托`EventArgs`**:如果想要传递更多信息,可以创建自定义的事件并在UI线程上触发。 ```csharp public class ValueUpdateArgs : EventArgs { public int NewValue { get; set; } public ValueUpdateArgs(int value) { NewValue = value; } } YourControl.ValueChanged += (sender, e) => { if (e is ValueUpdateArgs updateArgs) { YourControl.Value = updateArgs.NewValue; } }; // 在非UI线程上调用 new Thread(() => { RaiseValueChangedEvent(42); // 假设有一个 RaiseValueChangedEvent 的方法 }).Start(); ``` 3. **使用`TaskCompletionSource`**:对于一些长时间运行的操作,可以使用`TaskCompletionSource`结合`await`关键字,在完成任务后再更新UI。 ```csharp private TaskCompletionSource<int> tcs = new TaskCompletionSource<int>(); // 在非UI线程执行耗时操作 long newValue = DoSomeLongRunningTask(); tcs.SetResult(newValue); // 在UI线程监听完成信号并更新控件 tcs.Task.ContinueWith(_ => YourControl.Value = newValue); ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值