WPF后台线程更新UI

0、讲点废话

最近在做一个文件搜索的小软件,当文件多时,界面会出现假死的状况,于是乎想到另外开一个后台线程,更新界面上的ListView,但是却出现我下面的问题。

1、后台线程问题

2年前写过一个软件,里面后台线程是用如下方法:

1 Thread searchFileThread = new Thread(new ThreadStart(StartSearchFile));  //搜索文件后台线程
2 searchFileThread.IsBackground = true;
3 searchFileThread.Start();

虽然我将IsBackground属性置为true了,但是依然没有用,界面依然卡顿,不知为何,如果有大神路过,还请大神在评论区不吝赐教。最后上网baidu,google找了个别的办法,解决了它,效果还是很棒的。

2、解决界面假死问题

其实方法也很简单,微软早就想到此问题,提供了Dispather.Invoke()这一类方法。我将搜索到的文件放在lst_PaperList(这是一个Listview)中,StartSearchFile是我写的一个搜索文件的函数,然后用如下代码:

lst_PaperList.Dispatcher.BeginInvoke(DispatcherPriority.Background, new Action(StartSearchFile));

嗯,就这么写完了。但是这样子写完,好像没有点什么技术含量,我也在网上找了个不用另外开线程,实现更UI的方法,在此也分享一下。

3、不用线程实时更新UI

我的搜索软件中有个进度条,本来以为在循环让ProgressBar.Value++,就能更新ProgressBar的进度,没料到却不能,还是想得太简单。因此找了个不用自已开线程来更新ProgressBar.Value的方法,这个方法严格来讲不能说没有开线程,只是微软给我们做了。方法如下:

Step1 定义一个委托

private delegate void UpdateUIDelegate(System.Windows.DependencyProperty dp, Object value);

Step2 使用Dispatcher.Invoke方法

这里说一下这个方法其中一个重载的作用Invoke(Delegate, DispatcherPriority, Object[] )按指定的优先级并使用指定的参数在与 Dispatcher 关联的线程上同步执行指定的委托。更多的还是自行查MSDN。代码如下(其中pb_Search是我定义的一个进度条):

1 UpdateUIDelegate updatePbDelegate = new UpdateUIDelegate(pb_Search.SetValue);
2 
3 Dispatcher.Invoke(updatePbDelegate, System.Windows.Threading.DispatcherPriority.Background,
4                   new object[] { System.Windows.Controls.ProgressBar.ValueProperty, Convert.ToDouble(pb_Search.Value) });

完成了,就是这么简单粗暴。上面的代码不要生搬硬套,在适当的地方使用。

照葫芦画瓢,实时更新TextBlock的值

在WPF中,这个用绑定也行,但是效果可能没有这个好。代码如下(其中txt_SearchCount是用来显示搜索结果个数的):

1 UpdateUIDelegate updateTbDelegate = new UpdateUIDelegate(txt_SearchCount.SetValue);
2 
3 Dispatcher.Invoke(updateTbDelegate, System.Windows.Threading.DispatcherPriority.Background,
4                   new object[] { System.Windows.Controls.TextBlock.TextProperty, Convert.ToString(lst_EPaper.Count) });

 

posted @ 2018-07-15 17:48  EndlessCoding  阅读(1614)  评论(2编辑  收藏  举报