[C#] 走进异步编制程序的社会风气 – 在 GUI 中施行异步操作

走进异步编制程序的社会风气 – 在 GUI 中推行异步操作

【博主】反骨仔  【原来的小说地址】http://www.cnblogs.com/liqingwen/p/5877042.htmlca88官网, 

   这是继《[开头接触 async/await

异步编制程序](http://www.cnblogs.com/liqingwen/p/5831951.html)》、《[走进异步编程的世界

解析异步方法](http://www.cnblogs.com/liqingwen/p/5844095.html)》后的第三篇。主要介绍在
WinForm 中如何履行异步操作。

 

目录

 

一、在 WinForm 程序中实行异步操作

  上面通过窗体示例演示以下操作-点击按钮后:

    一将按钮禁止使用,并将标签内容改成:“Doing”(表示执行中);

    贰线程挂起3秒(模拟耗费时间操作);

    三启用按钮,将标签内容改为:“Complete”(表示执行到位)。

 1     public partial class Form1 : Form
 2     {
 3         public Form1()
 4         {
 5             InitializeComponent();
 6         }
 7 
 8         private void btnDo_Click(object sender, EventArgs e)
 9         {
10             btnDo.Enabled = false;
11             lblText.Text = @"Doing";
12 
13             Thread.Sleep(3000);
14 
15             btnDo.Enabled = true;
16             lblText.Text = @"Complete";
17         }
18     }

  然而执行结果却是:

ca88官网 1

图1-1

 

  【发现的题材】

    壹像样从没成为“Doing”?

    2并且拖动窗口的时候卡住不动了?

    33秒后突然变到想拖动到的地方?

    四并且文本变成“Complete”?

 

  【分析】GUI
程序在统一筹划中必要有所的显得变化都不可能不在主 GUI
线程中成功,如点击事件和活动窗体。Windows 程序时经过
新闻来兑现,消息放入音讯泵管理的新闻队列中。点击按钮时,按钮的Click新闻放入音信队列。音信泵从队列中移除该新闻,并初始拍卖点击事件的代码,即 btnDo_Click
事件的代码。

  btnDo_Click 事件会将触发行为的消息放入队列,但在 btnDo_Click
时间处理程序完全退出前(线程挂起 三 秒退出前),消息都心有余而力不足履行。(3
秒后)接着全体行为都发生了,但速度太快肉眼不可能分辨才没有意识标签改成“Doing”。

ca88官网 2

图壹-贰 点击事件

ca88官网 3

图1-三 点击事件实际实行进程

  

  今后大家投入 async/await 天性。

 1     public partial class Form1 : Form
 2     {
 3         public Form1()
 4         {
 5             InitializeComponent();
 6         }
 7 
 8         private async void btnDo_Click(object sender, EventArgs e)
 9         {
10             btnDo.Enabled = false;
11             lblText.Text = @"Doing";
12 
13             await Task.Delay(3000);
14 
15             btnDo.Enabled = true;
16             lblText.Text = @"Complete";
17         }
18     }

ca88官网 4

图1-4

  未来,正是本来希望看到的意义。

  【分析】btnDo_Click
事件处理程序先将前两条信息压入队列,然后将团结从总结机移出,在3秒后(等待空闲职分到位后
Task.Delay
)再将协调压入队列。那样能够保持响应,并保险全体的新闻能够在线程挂起的时光内被拍卖。

 

 1.1 Task.Yield

  Task.Yield 方法创立1个登时回到的
awaitable。等待3个Yield能够让异步方法在履行后续部分的同时重回到调用方法。能够将其领会为
离开当前消息队列,回到队列末尾,让 CPU 有时光处理任何职务。

 1     class Program
 2     {
 3         static void Main(string[] args)
 4         {
 5             const int num = 1000000;
 6             var t = DoStuff.Yield1000(num);
 7 
 8             Loop(num / 10);
 9             Loop(num / 10);
10             Loop(num / 10);
11 
12             Console.WriteLine($"Sum: {t.Result}");
13 
14             Console.Read();
15         }
16 
17         /// <summary>
18         /// 循环
19         /// </summary>
20         /// <param name="num"></param>
21         private static void Loop(int num)
22         {
23             for (var i = 0; i < num; i++) ;
24         }
25     }
26 
27     internal static class DoStuff
28     {
29         public static async Task<int> Yield1000(int n)
30         {
31             var sum = 0;
32             for (int i = 0; i < n; i++)
33             {
34                 sum += i;
35                 if (i % 1000 == 0)
36                 {
37                     await Task.Yield(); //创建异步产生当前上下文的等待任务
38                 }
39             }
40 
41             return sum;
42         }
43     }

ca88官网 5

 图1.1-1

  上述代码每执行1000次巡回就调用 Task.Yield
方法创造四个等候义务,让电脑有时光处理任何职责。该措施在 GUI
程序中是相比较有效的。

 

二、在 WinForm 中应用异步 Lambda 表明式

  将刚刚的窗口程序的点击事件稍微改变一下。

 1     public partial class Form1 : Form
 2     {
 3         public Form1()
 4         {
 5             InitializeComponent();
 6 
 7             //async (sender, e) 异步表达式
 8             btnDo.Click += async (sender, e) =>
 9             {
10                 Do(false, "Doing");
11 
12                 await Task.Delay(3000);
13 
14                 Do(true, "Finished");
15             };
16         }
17 
18         private void Do(bool isEnable, string text)
19         {
20             btnDo.Enabled = isEnable;
21             lblText.Text = text;
22         }
23     }

  照旧原先的配方,依然熟谙的意味,如故原来哪个窗口,变的只是内涵。

ca88官网 6

图2-1

 

三、贰个完好无损的 WinForm 程序

  今后在原本的基本功上添加了进程条,以及废除按钮。

 1     public partial class Form1 : Form
 2     {
 3         private CancellationTokenSource _source;
 4         private CancellationToken _token;
 5 
 6         public Form1()
 7         {
 8             InitializeComponent();
 9         }
10 
11         /// <summary>
12         /// Do 按钮事件
13         /// </summary>
14         /// <param name="sender"></param>
15         /// <param name="e"></param>
16         private async void btnDo_Click(object sender, EventArgs e)
17         {
18             btnDo.Enabled = false;
19 
20             _source = new CancellationTokenSource();
21             _token = _source.Token;
22 
23             var completedPercent = 0; //完成百分比
24             const int time = 10; //循环次数
25             const int timePercent = 100 / time; //进度条每次增加的进度值
26 
27             for (var i = 0; i < time; i++)
28             {
29                 if (_token.IsCancellationRequested)
30                 {
31                     break;
32                 }
33 
34                 try
35                 {
36                     await Task.Delay(500, _token);
37                     completedPercent = (i + 1) * timePercent;
38                 }
39                 catch (Exception)
40                 {
41                     completedPercent = i * timePercent;
42                 }
43                 finally
44                 {
45                     progressBar.Value = completedPercent;
46                 }
47             }
48 
49             var msg = _token.IsCancellationRequested ? $"进度为:{completedPercent}% 已被取消!" : $"已经完成";
50 
51             MessageBox.Show(msg, @"信息");
52 
53             progressBar.Value = 0;
54             InitTool();
55         }
56 
57         /// <summary>
58         /// 初始化窗体的工具控件
59         /// </summary>
60         private void InitTool()
61         {
62             progressBar.Value = 0;
63             btnDo.Enabled = true;
64             btnCancel.Enabled = true;
65         }
66 
67         /// <summary>
68         /// 取消事件
69         /// </summary>
70         /// <param name="sender"></param>
71         /// <param name="e"></param>
72         private void btnCancel_Click(object sender, EventArgs e)
73         {
74             if (btnDo.Enabled) return;
75 
76             btnCancel.Enabled = false;
77             _source.Cancel();
78         }
79     }

ca88官网 7

 图3-1

 

四、另一种异步格局 – BackgroundWorker 类

  与 async/await
分化的是,你有时候只怕要求一个拾1分的线程,在后台持续完毕某项职责,并时常与主线程通讯,那时就必要用到 BackgroundWorker
类。首要用于 GUI 程序。

  书中的万语千言比不上多少个简短的示范。

 1     public partial class Form2 : Form
 2     {
 3         private readonly BackgroundWorker _worker = new BackgroundWorker();
 4 
 5         public Form2()
 6         {
 7             InitializeComponent();
 8 
 9             //设置 BackgroundWorker 属性
10             _worker.WorkerReportsProgress = true;   //能否报告进度更新
11             _worker.WorkerSupportsCancellation = true;  //是否支持异步取消
12 
13             //连接 BackgroundWorker 对象的处理程序
14             _worker.DoWork += _worker_DoWork;   //开始执行后台操作时触发,即调用 BackgroundWorker.RunWorkerAsync 时触发
15             _worker.ProgressChanged += _worker_ProgressChanged; //调用 BackgroundWorker.ReportProgress(System.Int32) 时触发
16             _worker.RunWorkerCompleted += _worker_RunWorkerCompleted;   //当后台操作已完成、被取消或引发异常时触发
17         }
18 
19         /// <summary>
20         /// 当后台操作已完成、被取消或引发异常时发生
21         /// </summary>
22         /// <param name="sender"></param>
23         /// <param name="e"></param>
24         private void _worker_RunWorkerCompleted(object sender, RunWorkerCompletedEventArgs e)
25         {
26             MessageBox.Show(e.Cancelled ? $@"进程已被取消:{progressBar.Value}%" : $@"进程执行完成:{progressBar.Value}%");
27             progressBar.Value = 0;
28         }
29 
30         /// <summary>
31         /// 调用 BackgroundWorker.ReportProgress(System.Int32) 时发生
32         /// </summary>
33         /// <param name="sender"></param>
34         /// <param name="e"></param>
35         private void _worker_ProgressChanged(object sender, ProgressChangedEventArgs e)
36         {
37             progressBar.Value = e.ProgressPercentage;   //异步任务的进度百分比
38         }
39 
40         /// <summary>
41         /// 开始执行后台操作触发,即调用 BackgroundWorker.RunWorkerAsync 时发生
42         /// </summary>
43         /// <param name="sender"></param>
44         /// <param name="e"></param>
45         private static void _worker_DoWork(object sender, DoWorkEventArgs e)
46         {
47             var worker = sender as BackgroundWorker;
48             if (worker == null)
49             {
50                 return;
51             }
52 
53             for (var i = 0; i < 10; i++)
54             {
55                 //判断程序是否已请求取消后台操作
56                 if (worker.CancellationPending)
57                 {
58                     e.Cancel = true;
59                     break;
60                 }
61 
62                 worker.ReportProgress((i + 1) * 10);    //触发 BackgroundWorker.ProgressChanged 事件
63                 Thread.Sleep(250);  //线程挂起 250 毫秒
64             }
65         }
66 
67         private void btnDo_Click(object sender, EventArgs e)
68         {
69             //判断 BackgroundWorker 是否正在执行异步操作
70             if (!_worker.IsBusy)
71             {
72                 _worker.RunWorkerAsync();   //开始执行后台操作
73             }
74         }
75 
76         private void btnCancel_Click(object sender, EventArgs e)
77         {
78             _worker.CancelAsync();  //请求取消挂起的后台操作
79         }
80     }

ca88官网 8

图4-1

 

 传送门

  入门:《[走进异步编制程序的世界

  上篇:《走进异步编制程序的世界 –
剖析异步方法(上)
》《[走进异步编制程序的社会风气

 

 


【参考】《Illustrated C# 2012》

相关文章