C#巧妙使用关键字async/await

原文链接(程序杰杰):https://www.cnblogs.com/ningxinjie/p/12008561.html

 

经过一番的探索,终于搞清楚关键字async/await 在.net4.5之后可用的巧妙之处,在这里记录一下也与大家分享一下个人的心得体会

async:异步执行

await:异步执行中的等待其执行完(最大的优点是它是后台等待,因此不阻塞GUI,界面交互非常的好)

 

使用async方法要定义async Task或者async Task<T> 最好不要定义async void方法来调用,async void是处理程序等,总结结论就是要使用async Task或者async Task<T> 

使用asyn方法,可以用同步的格式,执行异步的代码,如下:

 1         int a = 0;
 2         private void button1_Click(object sender, EventArgs e)
 3         {
 4             Task.Run(()=> {
 5                 Stopwatch sw = new Stopwatch();
 6                 sw.Start();
 7                 Thread.Sleep(2000);
 8                 sw.Stop();
 9                 SetText(textBox1,this, sw.ElapsedMilliseconds.ToString());
10             });
11             var eee=Ce_async();//前面接收值,这样方法立即返回,主线程继续执行
12             //eee.IsCompleted
13             textBox3.Text = "主线程继续向下执行";
14 
15 
16         }
17         private async Task Ce_async()
18         {
19             await Task.Run(async ()=> {
20                 while (true)
21                 {
22                     await Task.Delay(100);//Thread.Sleep(2000);23                     a += 1;
24                     SetText(textBox2, this, a.ToString());
25                 }
26             });
27         }
28 
29 
30 
31         private delegate void SetTextdelegate(Control cr,Form f,string str);
32         private void SetText(Control cr, Form f, string str)
33         {
34             if (f.InvokeRequired) { f.Invoke(new SetTextdelegate(SetText), cr, f, str); }
35             else { cr.Text = str; }
36         }

正如上图第11行注释,使用接收值来接收async方法,该方法会立即返回值,主程序继续向下执行,该方法后台继续跑。

 

那么肯定有小伙伴与我当初有同样的疑惑,这样我执行用线程在这里执行不也可以吗?为什么要用async关键字呢,他到底有什么好处?

 

比如有一种情况:我的主线程如何有需要休眠等待的情况,那么直接使用Thread.Sleep(),会使界面卡死,如果此时我将方法修改为async,那么只需要await Task.Delay() 这样也会等待时间完成后再向下执行,但是它不会使主界面处于假死状态,如下:

 1    private async void button1_Click(object sender, EventArgs e)
 2         {
 3             await Task.Delay(1000);
 4             var aaa=Task.Run(() =>
 5             {
 6                 Stopwatch sw = new Stopwatch();
 7                 sw.Start();
 8                 Thread.Sleep(2000);
 9                 sw.Stop();
10                 SetText(textBox1, this, sw.ElapsedMilliseconds.ToString());
11             });
12             var eee=Ce_async();//前面接收值,这样方法立即返回,主线程继续执行
13             //eee.IsCompleted
14             textBox3.Text = "主线程继续向下执行";
15         }

 

再次重复一点,在async方法中使用接收值来接收Task,会使得Task不必加上await关键字而被迫使得主线程需要等待它执行完才能向下执行,接收值的目的就是立即得到返回值

使用接收值来接收Task会直接返回,代码继续向下执行,这一点很重要!!!

上图的第3行,如下:

await Task.Delay(1000);

其中await只能用在异步方法中,await会使得该线程等待await处的线程执行完,方可执行下方代码,正如该处代码所示,使用await Task.Delay(100); 代替 Thread.Sleep(100)

 

对于有返回值的这样来写:

Form1中按钮代码如下:

  private async void button1_Click(object sender, EventArgs e)
        {
           string a= await new Class1().Ceshifunc();
           MessageBox.Show(a);
        }

 

执行的Class1类方法为:

  public class Class1
    {
        private static readonly object obj = new object();
        public Task<String> Ceshifunc()
        {
            lock (obj)
            {
                var task = Task.Run(()=>
                {
                    Thread.Sleep(5000);
                    return "ok";
                });
                return task;
            }
        }
    }

 这样操作无卡顿,但是仍然无法实现lock内无法await,因为lock内的代码是task,他不会等待task执行完再返回,而是返回出去再执行,因此相当于没lock一样

 

我将所有代码放入Form1下

 

 

    private static readonly object obj = new object();
        public Task<String> Ceshifunc(string str, Form form, TextBox tb)
        {
            var task = Task.Run(() =>
            {
                lock (obj)
                {
                    string aa = label1.Text;
                    SetText(label1, this, aa + "1");
                    Thread.Sleep(2000);
                    SetText(tb, form, str);
                    return str;
                }
            });
            return task;
        }

        public Action<Control, Form, string> setText;
        public void SetText(Control tb, Form form, string str)
        {
            setText += SetText;
            if (form.InvokeRequired) { form.Invoke(setText, tb, form, str); }
            else { tb.Text = str; }
        }

 

按钮事件为:

  private async void button1_Click(object sender, EventArgs e)
        {
            string a = await Ceshifunc("b1", this, textBox1);
        }

 

 这样就可以实现了变相解决lock内的await使用

 

 

 

个人推荐另一个博客原文链接地址:https://www.cnblogs.com/mqxs/p/6550274.html

 

 以上就是个人的心得体会,(转载请注明)(程序杰杰)原文链接:https://www.cnblogs.com/ningxinjie/p/12008561.html



 

posted @ 2019-12-08 23:07  程序杰杰  阅读(1338)  评论(0编辑  收藏  举报