C#异步编程的三种模式

使用异步编程,方法调用是在后台运行(通常在线程和任务的帮助下),并且不会阻塞调用线程。
异步编程有三种模式:异步模式,基于事件的异步模式和基于任务的异步模式(TAP)。

一.异步模式

从.NET 1.0开始 .NET Framework就提供了异步特性,.NET Framework的许多类(但不是全部类)都实现了一个或多个异步模式,自定义类可以通过委托类型实现异步模式。
.NET Framework的许多类的异步模式定义了BeginXXX()方法和EndXXX方法。如HttpWebRequest类有一个同步方法GetResponse方法,其异步方法就是BeginGetResponse和EndGetResponse方法。BeginXXX()方法接受其同步方法的所有输入参数,EndXXX方法使用同步方法的所有输出参数,并按照同步方法的返回类型来返回结果。使用异步模式时,BeginXXX()方法还定义了一个AsyncCallback参数,用于接受在异步方法执行完成后调用的委托。BeginXXX()方法返回IAsyncResult,用于验证调用是否完成,并且一直等到方法的执行结束。

var client = new HttpWebRequest();
        client.BeginGetResponse( ar =>
        {
            client.EndGetResponse(ar);

        },null);

自定义类可以通过委托类型实现异步模式:
先编写一个同步示例:  

private void button1_Click(object sender, EventArgs e)
        {
            Thread.Sleep(5000);
            //MessageBox.Show("同步完成!");
        }

在窗体上点击该按钮时,线程睡眠五秒,五秒内不可以操作窗体。五秒之后才可以进行别的操作。
编写异步模式:  

private void button2_Click(object sender, EventArgs e)
        {
            Func<int,string> suncTest = (e1) =>
            {
                Thread.Sleep(e1);
                return "异步模式完成!";
            };

            suncTest.BeginInvoke(5000,ar =>
            {
                string re =  suncTest.EndInvoke(ar);
                MessageBox.Show(re);

            }, null);
        }

定义一个委托,并添加要执行的方法。然后调用BeginInvoke方法进行异步执行。BeginInvoke方法可以传递添加方法的参数,第一个参数是添加的方法的参数,第二个参数的类型是AsyncCallback。AsyncCallback是一个委托,需要IAsyncResult作为参数。当执行完异步方法之后,将调用这个委托引用的方法。使用suncTest.EndInvoke(ar)方法来检索结果。
这里不能直接把结果返回给UI,因为UI绑定到一个单独的线程,而回调方法在一个后台线程中运行。需要使用窗体的Invoke方法,它在会绑定到UI的集合中添加项。

Func<int,string> suncTest = (e1) =>
            {
                Thread.Sleep(e1);
                return "异步模式完成!";
            };
            string s = this.Invoke(suncTest, 5000).ToString();
            label1.Text = s;

二.基于事件的异步模式

在WF和WPF中,使用异步模式更新界面很复杂。.NET 2.0 推出基于事件的异步模式。在这种模式中,事件处理程序是被拥有同步上下文的线程调用,所以更新界面会很容易。这种模式也称为异步组件模式。
同样.NET Framework中的许多类提供了基于事件异步模式的方法,基于事件的异步模式定义了带有“Async”后缀的方法。

var client = new WebClient();
        client.Credentials = req.Credentials;
        client.DownloadStringCompleted += (sender1, e1) =>
          {
            string resp = e1.Result;
            var images = req.Parse(resp);
            foreach (var image in images)
            {
              searchInfo.List.Add(image);
            }
          };
        client.DownloadStringAsync(new Uri(req.Url));

对于自定义的类可以使用BackgroundWorker类(可以查看MSDN)来实现基于事件的异步模式:

private void button3_Click(object sender, EventArgs e)
            {
                string s="";
                BackgroundWorker bw = new BackgroundWorker();
                bw.DoWork += (sender1,e1) =>
                {
                    Thread.Sleep(5000);
                    s = "基于事件的异步完成!";
                };

                bw.RunWorkerCompleted += (sender1, e1) =>
                    {
                        label1.Text = s;
                    };

                bw.RunWorkerAsync();
            }

这里定义要调用的一个事件。当调用RunWorkerAsync()方法时,触发DoWork事件。当异步方法执行完成之后会触发RunWorkerCompleted事件,该事件会执行添加的方法。
这里可以直接访问UI元素,因为事件处理程序是从拥有上下文的线程中调用的,在WF和WPF应用程序中,拥有同步上下文的线程就是UI线程。

三.基于任务的异步模式(TAP)

在.NET 4.5中推出基于任务的异步模式。这种是基于.NET 4.0 中新增的Task类型,并通过async和await关键字来使用编译器功能。
同样.NET Framework中的许多类提供了基于事件异步模式的方法,基于事件的异步模式定义了带有“Async”后缀的方法,并返回一个Task类型。
这里介绍自定义的基于任务的异步模式

private async void button4_Click(object sender, EventArgs e)
        {

            string re =  await AsyncTaskTestAsync();
            label1.Text =re;
        }

        Task<string> AsyncTaskTestAsync()
        {
            return Task.Run(() =>
                {
                    Thread.Sleep(5000);
                    return "基于任务的异步完成!";
                }
            );
        }

基于任务的异步模式指定,在异步方法名后最好加上Async后缀,并返回一个任务。这里返回Task<string>。调用AsyncTaskTestAsync()时不需要声明一个Task<string>变量来设置AsyncTaskTestAsync()方法的返回结果。只需要声明一个string类型的变量,并使用await关键字。await关键字会接触线程(这里是UI线程)的阻塞,完成其它任务。当AsyncTaskTestAsync()方法完成之后,返回UI线程,UI线程就可以从后台线程中获得结果。然后执行await后面的代码。
使用await关键字需要用async修饰声明的方法。在AsyncTaskTestAsync方法完成前,该方法内的其它代码不会继续执行,但是启动button4_Click方法的线程可以被重用。
async只能用于返回Task或void的方法。不能用于程序的入口,即Main方法。await只能用于返回Task的方法。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • 深入分析C#异步编程详解

    同步方法和异步方法的区别同步方法调用在程序继续执行之前需要等待同步方法执行完毕返回结果异步方法则在被调用之后立即返回以便程序在被调用方法完成其任务的同时执行其它操作异步编程概览.NET Framework 允许您异步调用任何方法.定义与您需要调用的方法具有相同签名的委托:公共语言运行库将自动为该委托定义具有适当签名的 BeginInvoke 和 EndInvoke 方法.BeginInvoke 方法用于启动异步调用.它与您需要异步执行的方法具有相同的参数,只不过还有两个额外的参数(将在稍后描述)

  • C#中异步Socket通信编程代码实例

    本文将在C#中Socket同步通信的基础上,分析和研究Socket异步编程的实现方法,目的是深入了解Socket编程的基本原理,增强对网络游戏开发相关内容的认识. 什么是Socket编程的异步是实现 所谓Socket编程的异步实现是指按照异步过程来实现Socket编程,那么什么是异步过程呢,我们把在完成了一次调用后通过状态.通知和回调来告知调用者的方式成为异步过程,换句话说,在异步过程中当调用一个方法时,调用者并不能够立刻得到结果,只有当这个方法调用完毕后调用者才能获得调用结果.这样做的好处是什

  • c# 异步编程基础讲解

    现代应用程序广泛使用文件和网络 I/O.I/O 相关 API 传统上默认是阻塞的,导致用户体验和硬件利用率不佳,此类问题的学习和编码的难度也较大.而今基于 Task 的异步 API 和语言级异步编程模式颠覆了传统模式,使得异步编程非常简单,几乎没有新的概念需要学习. 异步代码有如下特点: 在等待 I/O 请求返回的过程中,通过让出线程来处理更多的服务器请求. 通过在等待 I/O 请求时让出线程进行 UI 交互,并将长期运行的工作过渡到其他 CPU,使用户界面的响应性更强. 许多较新的 .NET

  • C#异步编程几点需要注意的地方

    尽量不要编写返回值类型为void的异步方法 在通常情况下,建议大家不要编写那种返回值类型为void的异步方法,因为这样做会破坏该方法的启动者与方法本身之间的约定,这套约定本来可以确保主调方能够捕获到异步方法所发生的异常. 正常的异步方法是通过它返回的Task对象来汇报异常的.如果执行过程中发生了异常,那么Task对象就进入了faulted(故障)状态.主调方在对异步方法所返回的Task对象做await操作时,该对象若已处在faulted状态,系统则会将执行异步方法的过程中所发生的异常抛出,反之,

  • C#异步编程详解

    前言 本节主要介绍异步编程中Task.Async和Await的基础知识. 什么是异步? 异步处理不用阻塞当前线程来等待处理完成,而是允许后续操作,直至其它线程将处理完成,并回调通知此线程. 异步和多线程 相同点:避免调用线程阻塞,从而提高软件的可响应性. 不同点: 异步操作无须额外的线程负担,并且使用回调的方式进行处理,在设计良好的情况下,处理函数可以不必使用共享变量(即使无法完全不用,最起码可以减少 共享变量的数量),减少了死锁的可能.C#5.0 .NET4.5 以后关键字Async和Awai

  • C#实现异步编程的方法

    最近在我参与的几个.Net项目中都有用到异步编程,作为一名.Net小白,很有必要好好地学习一下C#异步编程. 什么是异步 异步指的就是不用阻塞当前线程来等待任务的完成,而是将任务扔到线程池中去执行,当前线程可以继续向下执行,直至其它线程将任务完成,并回调通知当前线程.整个任务从开始到结束都是异步完成的,不会阻塞当前线程.因此,异步很重要的一点就是,不会阻塞当前线程. 实现异步编程 在C#语言中,主要是通过委托来实现异步编程的,在委托类型中定义了两个方法BeginInvoke()和EndInvok

  • 基于C#的socket编程的TCP异步的实现代码

    一.摘要 本篇阐述基于TCP通信协议的异步实现. 二.实验平台 Visual Studio 2010 三.异步通信实现原理及常用方法 3.1 建立连接 在同步模式中,在服务器上使用Accept方法接入连接请求,而在客户端则使用Connect方法来连接服务器.相对地,在异步模式下,服务器可以使用BeginAccept方法和EndAccept方法来完成连接到客户端的任务,在客户端则通过BeginConnect方法和EndConnect方法来实现与服务器的连接. BeginAccept在异步方式下传入

  • c# 异步编程入门

    一.什么算异步?   广义来讲,两个工作流能同时进行就算异步,例如,CPU与外设之间的工作流就是异步的.在面向服务的系统中,各个子系统之间通信一般都是异步的,例如,订单系统与支付系统之间的通信是异步的,又如,在现实生活中,你去馆子吃饭,工作流是这样的,点菜->下单->做你的事->上菜->吃饭,这个也是异步的,具体来讲你和厨师之间是异步的,异步是如此重要,因外它代表者高效率(两者或两者以上的工作可以同时进行),但复杂,同步的世界简单,但效率极极低. 二.在编程中的异步   在编程中,

  • 解析C#多线程编程中异步多线程的实现及线程池的使用

    0.线程的本质 线程不是一个计算机硬件的功能,而是操作系统提供的一种逻辑功能,线程本质上是进程中一段并发运行的代码,所以线程需要操作系统投入CPU资源来运行和调度. 1.多线程: 使用多个处理句柄同时对多个任务进行控制处理的一种技术.据博主的理解,多线程就是该应用的主线程任命其他多个线程去协助它完成需要的功能,并且主线程和协助线程是完全独立进行的.不知道这样说好不好理解,后面慢慢在使用中会有更加详细的讲解. 2.多线程的使用: (1)最简单.最原始的使用方法:Thread oGetArgThre

  • c# 使用异步编程的方法

    怎么使用异步,就是用委托进行处理,如果委托对象在调用列表中只有一个方法,它就可以异步执行这个方法.委托类有两个方法,叫做BeginInvoke和EndInvoke,它们是用来异步执行使用. 异步有三种模式 等待模式,在发起了异步方法以及做了一些其它处理之后,原始线程就中断,并且等待异步方法完成之后再继续. 轮询模式,原始线程定期检查发起的线程是否完成,如果没有则可以继续做一些其它的事情. 回调模式,原始线程一直在执行,无需等待或检查发起的线程是否完成.在发起的线程中的引用方法完成之后,发起的线程

随机推荐