c#中使用async/await

可以先看之前关于task、Thread、backgroudwork的文章。

从性能和使用来说,Thread、threadPool、backgroudwork都弱于Task。使用task基本上都伴随着async/await,但并不是async/await都必须有task。async/await需要解决的问题就是,遇到耗时的任务,后台会进行等待,不卡界面操作,不卡界面是最主要的,等后台处理完成后,结果就会在界面上出来。本文接着task的例子进行。

之前界面业务回顾:

界面业务说明,label1是加法计算,分别有3个功能,对label1停止加法,暂停加法,继续加法。100 是减法计算

1.先看界面,说一下,label3是B线程开始标记, label4是B线程结束标记。之前的线程停止,暂停,继续依然能使用。

2.load中直接调用

3.AA方法,A方法不变和之前一样

      public async void AA()
        {
            await Task.Run(() =>
            {
                A();
            });
        }

       public void A()
        {
            for (int i = 0; i < 100; i++)
            {
                if (token.IsCancellationRequested)//这个是加法停止的判断
                {
                    return;
                }
                resetEvent.WaitOne();//相当于把resetEvent和线程task1进行绑定
                Thread.Sleep(1000);
                Invoke(new Action(() => label1.Text = i.ToString()));
            }

        }

 4.BB方法,B方法不变,和之前一样

    public async void BB()
        {
            label3.Text = "B线程开始";
            await Task.Run(() =>
            {
                B();
            });
            label4.Text = "B线程结束";
        }
   public void B()
        {
            for (int i = 0; i < 100; i++)
            {
                Thread.Sleep(1000);
                Invoke(new Action(() => label2.Text = (Convert.ToInt16(label2.Text) - 1).ToString()));
            }

        }

 5.效果

 代码

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Runtime.InteropServices;
using System.Text;
using System.Threading;
using System.Threading.Tasks;
using System.Windows.Forms;

namespace WindowsFormsApp1
{
    public partial class Form1 : Form
    {
        CancellationToken token;
        ManualResetEvent resetEvent = new ManualResetEvent(true);


        public Form1()
        {
            InitializeComponent();
        }

        private void Form1_Load(object sender, EventArgs e)
        {
            AA();
            BB();
        }

        public async void AA()
        {
            await Task.Run(() =>
            {
                A();
            });
        }

        public async void BB()
        {
            label3.Text = "B线程开始";
            await Task.Run(() =>
            {
                B();
            });
            label4.Text = "B线程结束";
        }

        public void A()
        {
            for (int i = 0; i < 100; i++)
            {
                if (token.IsCancellationRequested)//这个是加法停止的判断
                {
                    return;
                }
                resetEvent.WaitOne();//相当于把resetEvent和线程task1进行绑定
                Thread.Sleep(1000);
                Invoke(new Action(() => label1.Text = i.ToString()));
            }

        }

        public void B()
        {
            for (int i = 0; i < 20; i++)
            {
                Thread.Sleep(1000);
                Invoke(new Action(() => label2.Text = (Convert.ToInt16(label2.Text) - 1).ToString()));
            }

        }

        /// <summary>
        /// 停止线程
        /// </summary>
        /// <param name="sender"></param>
        /// <param name="e"></param>
        private void button1_Click(object sender, EventArgs e)
        {
            var tokenSource = new CancellationTokenSource();
            token = tokenSource.Token;
            tokenSource.Cancel();
        }
        /// <summary>
        /// 继续
        /// </summary>
        /// <param name="sender"></param>
        /// <param name="e"></param>
        private void button2_Click(object sender, EventArgs e)
        {
            resetEvent.Set();
        }

        /// <summary>
        /// 暂停
        /// </summary>
        /// <param name="sender"></param>
        /// <param name="e"></param>
        private void button3_Click(object sender, EventArgs e)
        {
            resetEvent.Reset();
        }
    }
}

 拓展

        //无返回值
        public async void A()
        {
            var v = await File.ReadAllBytesAsync("1.txt");
        }
        //有返回值
        public async Task<int> A1()
        {
            var v = await File.ReadAllBytesAsync("1.txt");
            return v.Length;
        }
        //事件上使用
        private async void B_Click(object sender, RoutedEventArgs e)
        {
            await Task.Run(() =>
           {
               File.ReadAllText("1.txt");
           });
        }
        //main上必须加Task
        static async Task Main(string[] args)
        {
            var v = await File.ReadAllBytesAsync("1.txt");
        }
        //返回IAsyncResult
        public static async Task<IAsyncResult> A2()
        {
            var v = await File.ReadAllTextAsync("1.txt");
            return Task.FromResult(v);
        }

Task的方法 

            Task task1 = Task.Run(() =>
            {
                A();
            });
            Task task2 = Task.Run(() =>
            {
                B();
            });
            Task.WhenAll(task1, task2);   //全部执行完成
            Task.WaitAny(task1, task2);   //其中一个执行完成

按钮异步 

        private async void button1_Click(object sender, EventArgs e)
        {
            var t = Task.Run(() => 
            {
                Thread.Sleep(4000);
                return "123";
            }
            );
            label1.Text = await t;
        }

实际应用:让一个功能先运行5秒,然后再继续运行。

有2种写法,效果一样

第一种,就是常规写法。

第二种使用了“技巧”,看起来,代码更加的清晰整洁。如果有实际应用,推荐第二种写法。

     private async void button1_Click(object sender, EventArgs e)
        {
            await Task.Run(() =>
              {
                  Thread.Sleep(5000);
              });
            button2.Text = "666";   //需要加async

            //Task.Run(() =>
            //{
            //    Thread.Sleep(5000);
            //}).GetAwaiter().OnCompleted(() =>
            //{
            //    Invoke(new Action(() => button2.Text = "666"));      //不需要加async
            //});
        }

必须带async,必须带await

必须带async,必须带await

必须带async,必须带await

await后面必须直接带异步的方法,例如ReadAllTextAsync;或者间接的带异步方法,也就是匿名方法的表示,例如Task.Run(()=>{})。

来源:c#中使用async/await_c# async awari-CSDN博客

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

故里2130

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值