wpf线程中更新UI的4种方式

在wpf中,更新UI上面的数据,那是必经之路,搞不好,就是死锁,或者没反应,很多时候,都是嵌套的非常深导致的。但是更新UI的方式,有很多的种,不同的方式,表示的意思不一样,但是眼睛看到的,似乎是一回事。

首先我们创建一个简单的wpf程序

业务就是,一直点击确定,然后更新数据即可,比较简单,通过简单的案例来了解一个wpf中更新UI的4种方法。

第一种:

点击确定后,界面先变化Hello WPF11,再变化Hello WPF12,并且界面可以任意拖动不卡。

这个方法是全局性质的。

cs 复制代码
using System.Windows;

namespace WpfApp6Demo
{
    /// <summary>
    /// Interaction logic for MainWindow.xaml
    /// </summary>
    public partial class MainWindow : Window
    {
        public MainWindow()
        {
            InitializeComponent();
        }

        private async void btnLogin_Click(object sender, RoutedEventArgs e)
        {
            await Task.Run(() =>
            {
                // 耗时操作
                Thread.Sleep(2000);
                UpdateTextBlock("Hello WPF11");
            });

            await Task.Run(() =>
            {
                // 耗时操作
                Thread.Sleep(2000);
                UpdateTextBlock("Hello WPF12");
            });
        }


        private void UpdateTextBlock(string text)
        {
            System.Windows.Application.Current.Dispatcher.Invoke(new Action(() =>
            {
                txtUsername.Text = text;
            }));
        }

    }
}

第二种:

点击确定后,和前面的效果是一样的。

这个方法是当前界面性质的。

cs 复制代码
using System.Windows;

namespace WpfApp6Demo
{
    /// <summary>
    /// Interaction logic for MainWindow.xaml
    /// </summary>
    public partial class MainWindow : Window
    {
        public MainWindow()
        {
            InitializeComponent();
        }

        private async void btnLogin_Click(object sender, RoutedEventArgs e)
        {
            await Task.Run(() =>
            {
                // 耗时操作
                Thread.Sleep(2000);
                UpdateTextBlock("Hello WPF11");
            });

            await Task.Run(() =>
            {
                // 耗时操作
                Thread.Sleep(2000);
                UpdateTextBlock("Hello WPF12");
            });
        }


        private void UpdateTextBlock(string text)
        {
            //System.Windows.Application.Current.Dispatcher.Invoke(new Action(() =>
            //{
            //    txtUsername.Text = text;
            //}));

            this.Dispatcher.Invoke(new Action(() =>
             {
                 txtUsername.Text = text;
             }));
        }

    }
}

第三种:

点击确定后,和前面的效果是一样的。

这个方法是当前控件性质的。

cs 复制代码
using System.Windows;

namespace WpfApp6Demo
{
    /// <summary>
    /// Interaction logic for MainWindow.xaml
    /// </summary>
    public partial class MainWindow : Window
    {
        public MainWindow()
        {
            InitializeComponent();
        }

        private async void btnLogin_Click(object sender, RoutedEventArgs e)
        {
            await Task.Run(() =>
            {
                // 耗时操作
                Thread.Sleep(2000);
                UpdateTextBlock("Hello WPF11");
            });

            await Task.Run(() =>
            {
                // 耗时操作
                Thread.Sleep(2000);
                UpdateTextBlock("Hello WPF12");
            });
        }


        private void UpdateTextBlock(string text)
        {
            //System.Windows.Application.Current.Dispatcher.Invoke(new Action(() =>
            //{
            //    txtUsername.Text = text;
            //}));


            //this.Dispatcher.Invoke(new Action(() =>
            // {
            //     txtUsername.Text = text;
            // }));



            txtUsername.Dispatcher.Invoke(new Action(() =>
            {
                txtUsername.Text = text;
            }));

        }

    }
}

第四种:

点击确定后,和前面的效果是一样的。

这个方法是当前控件性质的,但是还对当前控件进行是否有访问权限进行了判断。

cs 复制代码
using System.Windows;

namespace WpfApp6Demo
{
    /// <summary>
    /// Interaction logic for MainWindow.xaml
    /// </summary>
    public partial class MainWindow : Window
    {
        public MainWindow()
        {
            InitializeComponent();
        }

        private async void btnLogin_Click(object sender, RoutedEventArgs e)
        {
            await Task.Run(() =>
            {
                // 耗时操作
                Thread.Sleep(2000);
                UpdateTextBlock("Hello WPF11");
            });

            await Task.Run(() =>
            {
                // 耗时操作
                Thread.Sleep(2000);
                UpdateTextBlock("Hello WPF12");
            });
        }


        private void UpdateTextBlock(string text)
        {
            //System.Windows.Application.Current.Dispatcher.Invoke(new Action(() =>
            //{
            //    txtUsername.Text = text;
            //}));


            //this.Dispatcher.Invoke(new Action(() =>
            // {
            //     txtUsername.Text = text;
            // }));



            //txtUsername.Dispatcher.Invoke(new Action(() =>
            //{
            //    txtUsername.Text = text;
            //}));

            if (!txtUsername.Dispatcher.CheckAccess())
            {
                txtUsername.Dispatcher.Invoke(new Action<string>(UpdateTextBlock), text);
            }
            else
            {
                txtUsername.Text = text;
            }

        }

    }
}

以上都是以同步Invoke的方式进行调用的,异步的话使用BeginInvoke。

总结:通过案例,可以了解到,上面4种方式,对于此案例来说都可以达到最终的效果,但是那一种效果最好,并没有体现出来。博主认为:第三种最好,原因是从这个需求上考虑的,因为需求需要更新的就是txtUsername上面的数据,那么直接作用于它,对于资源的耗损,将是最少得。可能有人认为第四种方式最好,进行了线程判断,似乎更加的安全,那么您认为呢???

本文来源:

wpf线程中更新UI的4种方式-CSDN博客

相关推荐
yuegu7778 分钟前
DevUI Modal 模态弹窗组件使用详解
ui·前端框架
少年张二狗2 小时前
Vue + Element-UI 图片上传实现拖拽排序功能
前端·vue.js·ui
sailing-data3 小时前
【UI Qt】入门笔记
开发语言·qt·ui
L、2185 小时前
Flutter + OpenHarmony 分布式能力融合:实现跨设备 UI 共享与协同控制(终极篇)
javascript·分布式·flutter·ui·智能手机·harmonyos
鸿蒙开发工程师—阿辉6 小时前
HarmonyOS 5 极致动效实验室:给 UI 注入“物理动效”
ui·华为·harmonyos
lin62534221 天前
Android右滑解锁UI,带背景流动渐变动画效果
android·ui
hnlgzb1 天前
material3和xml的UI会相差很大么?
xml·ui
kylinmin1 天前
卸载微软电脑管家:一次性彻底移除
前端·ui·xhtml
Wiktok1 天前
【WIT】解决导入pywinauto相关库会导致程序UI界面(tkinter/pyside6)浏览文件等操作卡住问题
python·ui·pywinauto