.NET1.0版本中的异步编程模型(APM)

一、概念

.NET 1.0提出了APM(Asynchronous Programming Model)即异步编程模式。

.NET的类库有以BeginXXX和EndXXX类似的方法,就是使用异步编程模型。

NET Framework很多类也实现了该模式,同时我们也可以自定义类来实现该模式,即在自定义的类中实现返回类型为IAsyncResult接口的BeginXXX方法和EndXXX方法,另外委托类型也定义了BeginInvoke和EndInvoke方法。

异步编程模型的本质

利用委托和线程池帮助我们实现异步编程模型模式。

该模式利用一个线程池线程去执行一个操作,在FileStream类BeginRead方法中就是执行一个读取文件操作,该线程池线程会立即将控制权返回给调用线程,此时线程池线程在后台进行这个异步操作;

异步操作完成之后,通过回调函数来获取异步操作返回的结果,此时就是利用委托的机制。

1、BeginXxx方法——开始执行异步操作介绍

当需要读取文件中的内容时,我们通常会采用FileStream的同步方法Read来读取,该同步方法的定义为:

// 从文件流中读取字节块并将该数据写入给定的字节数组中
public override int Read(byte[] array, int offset, int count )

该同步方法会堵塞执行的线程。

可以通过BeginRead方法来实现异步编程,使读取操作不再堵塞UI线程。BeginRead方法代表异步执行Read操作,并返回实现IAsyncResult接口的对象,该对象存储着异步操作的信息。

// 开始异步读操作
// 前面的3个参数和同步方法代表的意思一样,这里就不说了,可以看到这里多出了2个参数
// userCallback代表当异步IO操作完成时,你希望由一个线程池线程执行的方法,该方法必须匹配AsyncCallback委托
// stateObject代表你希望转发给回调方法的一个对象的引用,在回调方法中,可以查询IAsyncResult接口的AsyncState属性来访问该对象
public override IAsyncResult BeginRead(byte[] array, int offset, int numBytes, AsyncCallback userCallback, Object stateObject)

从上面的代码中可以看出异步方法和同步方法的区别,如果你在使用该异步方法时,不希望异步操作完成后调用任何代码,你可以把userCallback参数设置为null

2、EndXxx方法——结束异步操作介绍

前面介绍完了BeginXxx方法,我们看到所有BeginXxx方法返回的都是实现了IAsyncResult接口的一个对象,并不是对应的同步方法那样直接得到结果。

此时我们需要调用对应的EndXxx方法来结束异步操作,并向该方法传递IAsyncResult对象,EndXxx方法的返回类型就是和同步方法一样的。例如,FileStreamEndRead方法返回一个Int32来代表从文件流中实际读取的字节数。

// 摘要:  等待挂起的异步读取完成。
// asyncResult:对要完成的挂起异步请求的引用。
// 返回结果: 从流中读取的字节数.
public virtual int EndRead(IAsyncResult asyncResult);

对于访问异步操作的结果,APM的首选方式是:
使用 AsyncCallback委托来指定操作完成时要调用的方法,在操作完成后调用的方法中调用EndXxx操作来获得异步操作的结果。

二、APM示例:

代码:

private static void Main(string[] args)
{
    string downUrl = "http://download.microsoft.com/download/5/B/9/5B924336-AA5D-4903-95A0-56C6336E32C9/TAP.docx";
    DownLoadFileSync(downUrl);  //同步下载文件,在下载操作完成之后我们才可以看到"Start DownLoad File......." 消息显示
    DownloadFileAsync(downUrl); //异步下载文件,在下载操作完成之前我们就可以看到"Start DownLoad File......." 消息显示
    Console.WriteLine("开始下载文件.........");
    Console.ReadLine();
}

//同步下载文件
private static void DownLoadFileSync(string url)
{
    RequestState req = new RequestState(); // 创建一个 RequestState 实例
    try
    {
        HttpWebRequest myHttpWebRequest = (HttpWebRequest)WebRequest.Create(url);   // 初始化一个  HttpWebRequest 对象
        req.request = myHttpWebRequest; // 指派 HttpWebRequest实例到requestState的request字段.
        req.response = (HttpWebResponse)myHttpWebRequest.GetResponse();
        req.streamResponse = req.response.GetResponseStream();

        int readSize = req.streamResponse.Read(req.BufferRead, 0, req.BufferRead.Length);
        while (readSize > 0)
        {
            req.filestream.Write(req.BufferRead, 0, readSize);
            readSize = req.streamResponse.Read(req.BufferRead, 0, req.BufferRead.Length);
        }

        Console.WriteLine("\n此文件的长度是: {0}", req.filestream.Length);
        Console.WriteLine("下载完成,下载路径是: {0}", req.savepath);
    }
    catch (Exception e)
    {
        Console.WriteLine("错误信息:{0}", e.Message);
    }
    finally
    {
        req.response.Close();
        req.filestream.Close();
    }
}

//异步下载文件
private static void DownloadFileAsync(string url)
{
    try
    {
        HttpWebRequest myHttpWebRequest = (HttpWebRequest)WebRequest.Create(url);
        RequestState req = new RequestState();
        req.request = myHttpWebRequest;
        myHttpWebRequest.BeginGetResponse(new AsyncCallback(ResponseCallback), req);
    }
    catch (Exception e)
    {
        Console.WriteLine("错误信息:{0}", e.Message);
    }
}

//每个异步操作完成时,将调用下面的方法
private static void ResponseCallback(IAsyncResult callbackresult)
{
    RequestState req = (RequestState)callbackresult.AsyncState; // 获取 RequestState 对象
    HttpWebRequest myHttpRequest = req.request;
    req.response = (HttpWebResponse)myHttpRequest.EndGetResponse(callbackresult); // 结束一个对英特网资源的的异步请求
    Stream responseStream = req.response.GetResponseStream(); //从服务器获取响应流
    req.streamResponse = responseStream;
    IAsyncResult asynchronousRead = responseStream.BeginRead(req.BufferRead, 0, req.BufferRead.Length, ReadCallBack, req);//异步读取流到字节数组
}

// 写字节数组到 FileStream
private static void ReadCallBack(IAsyncResult asyncResult)
{
    try
    {
        RequestState req = (RequestState)asyncResult.AsyncState; // 获取 RequestState 对象
        Stream responserStream = req.streamResponse;   //从RequestState对象中获取 Response Stream

        int readSize = responserStream.EndRead(asyncResult);
        if (readSize > 0)
        {
            req.filestream.Write(req.BufferRead, 0, readSize);
            responserStream.BeginRead(req.BufferRead, 0, req.BufferRead.Length, ReadCallBack, req);//循环调用ReadCallBack方法。
        }
        else
        {
            Console.WriteLine("\n此文件的长度是: {0}", req.filestream.Length);
            Console.WriteLine("下载完成,下载路径是: {0}", req.savepath);
            req.response.Close();
            req.filestream.Close();
        }
    }
    catch (Exception e)
    {
        Console.WriteLine("错误信息:{0}", e.Message);
    }
}

//存储请求的状态类
public class RequestState
{
    public int BufferSize = 1024;
    public string savepath = Environment.GetFolderPath(Environment.SpecialFolder.Desktop) + "\\TAP.docx";
    public byte[] BufferRead;
    public HttpWebRequest request;
    public HttpWebResponse response;
    public Stream streamResponse;
    public FileStream filestream;

    public RequestState()
    {
        BufferRead = new byte[BufferSize];
        request = null;
        streamResponse = null;
        if (File.Exists(savepath))
        {
            File.Delete(savepath);
        }
        filestream = new FileStream(savepath, FileMode.OpenOrCreate);
    }
}

运行结果:

使用同步方法下载文件的运行结果为:(从运行结果也可以看出,调用的是同步方法时,此时会堵塞主线程,直到文件的下载操作被完成之后主线程才继续执行后面的代码)

使用APM异步编程:运行结果为(从运行结果也可以看出,在主线程中调用 DownloadFileAsync(downUrl)方法时,DownloadFileAsync(downUrl)方法中的req.BeginGetResponse调用被没有阻塞调用线程(即主线程),而是立即返回到主线程,是主线程后面的代码可以立即执行)

三、委托实例的异步调用(BeginInvoke、EndInvoke方法

在前面的介绍中已经提到委托类型也会定义了BeginInvoke方法和EndInvoke方法,所以委托类型也实现了异步编程模型,所以可以使用委托的BeginInvokeEndInvoke方法来回调同步方法从而实现异步编程。

因为调用委托的BeginInvoke方法来执行一个同步方法时,此时会使用线程池线程回调这个同步方法并立即返回到调用线程中,由于耗时操作在另外一个线程上运行,所以执行BeginInvoke方法的主线程就不会被堵塞。

下面实现在线程池线程中如何更新GUI线程中窗体。

// 定义用来实现异步编程的委托
private delegate string AsyncMethodCaller(string fileurl);

private void btnDownLoad_Click(object sender, EventArgs e)
{
    AsyncMethodCaller methodCaller = new AsyncMethodCaller(DownLoadFileSync);//DownLoadFileSync为同步下载文件的方法
    methodCaller.BeginInvoke(this.txbUrl.Text.Trim(), GetResult, null);
}

// 异步操作完成时执行的方法
private void GetResult(IAsyncResult result)
{
    AsyncMethodCaller caller = (AsyncMethodCaller)((AsyncResult)result).AsyncDelegate;// 或者(AsyncMethodCaller)result.AsyncState;
    string returnstring = caller.EndInvoke(result); // 调用EndInvoke去等待异步调用完成并且获得返回值,如果异步调用尚未完成,则 EndInvoke 会一直阻止调用线程,直到异步调用完成

    // 然后Invoke方法来使更新GUI操作方法由GUI 线程去执行
    this.Invoke(new MethodInvoker(() =>
    {
        rtbState.Text = returnstring;
        btnDownLoad.Enabled = true;
    }));
}

运行的结果为:

上例子中使用了控件的Invoke方式进行异步回调访问控件的方法,其背后是通过获得GUI线程的同步上文对象SynchronizationContext,然后同步调用同步上下文对象的post方法把要调用的方法交给GUI线程去处理。

四、task实例

假如现在有这样的一个需求,我们需要从3个txt文件中读取字符,然后进行倒序,前提是不能阻塞主线程。

如果不用task的话我可能会用工作线程去监视一个bool变量来判断文件是否全部读取完毕,然后再进行倒序,我也说了,相对task来说还是比较麻烦的,这里我就用task来实现。

private static void Main()
{
    string[] array = { "C://1.txt", "C://2.txt", "C://3.txt" };
    List<Task<string>> taskList = new List<Task<string>>(3);
    foreach (var item in array)
    {
        taskList.Add(ReadAsyc(item));
    }
    Task.Factory.ContinueWhenAll(taskList.ToArray(), i =>
    {
        string result = string.Empty;
        //获取各个task返回的结果
        foreach (var item in i)
        {
            result += item.Result;
        }
        //倒序
        String content = new String(result.OrderByDescending(j => j).ToArray());
        Console.WriteLine("倒序结果:" + content);
    });
    Console.WriteLine("我是主线程,我不会被阻塞");
    Console.ReadKey();
}

//异步读取
private static Task<string> ReadAsyc(string path)
{
    FileInfo info = new FileInfo(path);
    byte[] b = new byte[info.Length];
    FileStream fs = new FileStream(path, FileMode.Open);
    Task<int> task = Task<int>.Factory.FromAsync(fs.BeginRead, fs.EndRead, b, 0, b.Length, null, TaskCreationOptions.None);
    //返回当前task的执行结果
    return task.ContinueWith(i =>
    {
        return i.Result > 0 ? Encoding.Default.GetString(b) : string.Empty;
    }, TaskContinuationOptions.ExecuteSynchronously);
}

到此这篇关于.NET1.0异步编程模型(APM)的文章就介绍到这了。希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • .net4.5使用async和await异步编程实例

    关于异步编程的简单理解: 在.NET4.5中新增了异步编程的新特性async和await,使得异步编程更为简单.通过特性可以将这项复杂的工作交给编译器来完成了.之前传统的方式来实现异步编程较为复杂,这样对于程序猿来说处理起来比较困难,调试也没那么方便,后续的维护工作也比较痛苦. Async和Await关键字是C#异步编程的核心.通过使用这两个关键字,你可以使用.NET Framework 或 Windows Runtime的资源创建一个异步方法如同创建一个同步方法一样容易. 接下来通过VS201

  • 在.NET Core中使用异步编程的方法步骤

    近期对于异步和多线程编程有些启发,所以我决定把自己的理解写下来. 思考:为什么要使用异步编程? 我们先看看同步方法和异步方法之前在程序中执行的逻辑: 1. 同步方法 static void Main(string[] args) { Console.WriteLine($"{DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss ms")}:开始"); // 调用同步方法 SyncTestMethod(); Console.WriteL

  • .NET中的异步编程-EAP/APM使用方法及案例介绍

    从.NET 4.5开始,支持的三种异步编程模式: •基于事件的异步编程设计模式 (EAP,Event-based Asynchronous Pattern) •异步编程模型(APM,Asynchronous Programming Model) •基于任务的编程模型(TAP,Task-based Asynchronous Pattern) 基于任务的异步模式 (TAP) 是基于 System.Threading.Tasks 命名空间的 Task 和 Task<TResult>,用于表示任意异步

  • .NET实现异步编程async和await

    await和async是.NET Framework4.5框架.C#5.0语法里面出现的,await和async是语法糖. 注意: 1.async出现在方法的声明里面,任何一个方法都可以增加async. 2.await放在Task前面,async和await是成对出现的,只有async是没有意义的,只有await是报错的. 只有async是没有意义的. 只有await是报错的. 3.await 只能放在task前面,不推荐void返回值,使用Task来代替.Task和Task<T>能够使用aw

  • ASP.Net中的async+await异步编程的实现

    在.NET Framework4.5框架.C#5.0语法中,通过async和await两个关键字,引入了一种新的基于任务的异步编程模型(TAP).在这种方式下,可以通过类似同步方式编写异步代码,极大简化了异步编程模型. 用法: public async Task<int> GetDotNetCountAsync() { // Suspends GetDotNetCount() to allow the caller (the web server) // to accept another r

  • .NET异步编程模式的三种类型介绍

    一.引言 .NET中很多的类.接口在设计的时候都考虑了多线程问题,简化了多线程程序的开发,不用自己去写WaitHandler等这些底层的代码,由于历史的发展,这些类的接口设计有着三种不同的风格:EAP.APM和TPL.目前重点用TPL. 二.EAP EAP是Event-based Asynchronous Pattem(基于事件的异步模型)的简写,类似于Ajax中的XmlHttpRequest,send之后并不是处理完成了,而是在onreadystatechange事件中再通知处理完成.看下面的

  • .NET异步编程总结----四种实现模式代码总结

    最近很忙,既要外出找工作又要兼顾老板公司的项目.今天在公司,忙里偷闲,总结一下.NET中的异步调用函数的实现方法,DebugLZQ在写这篇博文之前自己先动手写了本文的所有示例代码,开写之前是做过功课的,用代码说话方有说服力. 本文的内容旨在用最简洁的代码来把异步调用的方法说清楚,园子里的高手老鸟可以绕行,不喜勿喷,非诚勿扰~ lz的前一篇文章简单的说了下异步,主要是从理解上来讲:这篇文章主要写具体的实现方法.实现异步编程有4种方法可供选择,这4种访求实际上也对应着4种异步调用的模式,分为"等待&

  • .NET1.0版本中的异步编程模型(APM)

    一.概念 .NET 1.0提出了APM(Asynchronous Programming Model)即异步编程模式. .NET的类库有以BeginXXX和EndXXX类似的方法,就是使用异步编程模型. NET Framework很多类也实现了该模式,同时我们也可以自定义类来实现该模式,即在自定义的类中实现返回类型为IAsyncResult接口的BeginXXX方法和EndXXX方法,另外委托类型也定义了BeginInvoke和EndInvoke方法. 异步编程模型的本质 利用委托和线程池帮助我

  • C#5.0中的异步编程关键字async和await

    一.Asynchronous methods 异步方法 .NET 4.5 的推出,对于C#又有了新特性的增加——就是C#5.0中async和await两个关键字,这两个关键字简化了异步编程. 使用async修饰的方法被称为异步方法,这个方法调用时应该在前面加上await. 异步方法命名应该以Async结尾,这样大家知道调用的时候使用await. async和await关键字只是编译器的功能,编译器最终会用Task类创建代码. 1.创建返回任务的异步方法 建立一个同步方法Greeting,该方法在

  • .NET2.0版本中基于事件的异步编程模式(EAP)

    一.引言 APM为我们实现异步编程提供了一定的支持,同时它也存在着一些明显的问题——不支持对异步操作的取消和没有提供对进度报告的功能,对于有界面的应用程序来说,进度报告和取消操作的支持也是必不可少的. 微软在.NET 2.0的时候就为我们提供了一个新的异步编程模型,也就是基于事件的异步编程模型——EAP(Event-based Asynchronous Pattern ). 二.介绍 实现了基于事件的异步模式的类将具有一个或者多个以Async为后缀的方法和对应的Completed事件,并且这些类

  • python中asyncio异步编程学习

    1.   想学asyncio,得先了解协程 携程的意义: 计算型的操作,利用协程来回切换执行,没有任何意义,来回切换并保存状态 反倒会降低性能. IO型的操作,利用协程在IO等待时间就去切换执行其他任务,当IO操作结束后再自动回调,那么就会大大节省资源并提供性能,从而实现异步编程(不等待任务结束就可以去执行其他代码 2.协程和多线程之间的共同点和区别: 共同点: 都是并发操作,多线程同一时间点只能有一个线程在执行,协程同一时间点只能有一个任务在执行: 不同点: 多线程,是在I/O阻塞时通过切换线

  • nodejs中的异步编程知识点详解

    简介 因为javascript默认情况下是单线程的,这意味着代码不能创建新的线程来并行执行.但是对于最开始在浏览器中运行的javascript来说,单线程的同步执行环境显然无法满足页面点击,鼠标移动这些响应用户的功能.于是浏览器实现了一组API,可以让javascript以回调的方式来异步响应页面的请求事件. 更进一步,nodejs引入了非阻塞的 I/O ,从而将异步的概念扩展到了文件访问.网络调用等. 今天,我们将会深入的探讨一下各种异步编程的优缺点和发展趋势. 同步异步和阻塞非阻塞 在讨论n

  • 一文秒懂nodejs中的异步编程

    文章目录 简介同步异步和阻塞非阻塞javascript中的回调回调函数的错误处理回调地狱 ES6中的Promise什么是PromisePromise的特点Promise的优点Promise的缺点Promise的用法Promise的执行顺序 async和awaitasync的执行顺序async的特点 总结 简介 因为javascript默认情况下是单线程的,这意味着代码不能创建新的线程来并行执行.但是对于最开始在浏览器中运行的javascript来说,单线程的同步执行环境显然无法满足页面点击,鼠标

  • 浅谈node.js中async异步编程

    1.什么是异步编程? 异步编程是指由于异步I/O等因素,无法同步获得执行结果时, 在回调函数中进行下一步操作的代码编写风格,常见的如setTimeout函数.ajax请求等等. 示例: for (var i = 1; i <= 3; i++) { setTimeout(function(){ console.log(i); }, 0); }; 这里大部分人会认为输出123,或者333.其实它会输出 444 这里就是我们要说的异步编程了. 高级函数的定义 这里为什么会说到高级函数,因为高级函数是异

  • Javascript中的异步编程规范Promises/A详细介绍

    Javascript里异步编程逐渐被大家接受,先前大家一般通过回调嵌套,setTimeout.setInterval等方式实现,代码看起来非常不直观,不看整个代码逻辑很难快速理解.Javascript里异步函数大概有I/O函数(Ajax.postMessage.img load.script load等).计时函数(setTimeout.setInterval)等. 这些我们都很熟悉,在复杂的应用中往往会嵌套多层,甚至以为某些步骤未完成而导致程序异常,最简单的例子:比如你往DOM中注入节点,你必

随机推荐