广告
返回顶部
首页 > 资讯 > 后端开发 > ASP.NET >.NET1.0版本中的异步编程模型(APM)
  • 954
分享到

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

2024-04-02 19:04:59 954人浏览 独家记忆
摘要

一、概念 .net 1.0提出了APM(Asynchronous Programming Model)即异步编程模式。 .NET的类库有以BeginXXX和EndXXX类似的方法,就

一、概念

.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)的文章就介绍到这了。希望对大家的学习有所帮助,也希望大家多多支持编程网。

--结束END--

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

本文链接: https://www.lsjlt.com/news/150927.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • .NET1.0版本中的异步编程模型(APM)
    一、概念 .NET 1.0提出了APM(Asynchronous Programming Model)即异步编程模式。 .NET的类库有以BeginXXX和EndXXX类似的方法,就...
    99+
    2022-11-13
  • .NET2.0版本中基于事件的异步编程模式(EAP)
    一、引言 APM为我们实现异步编程提供了一定的支持,同时它也存在着一些明显的问题——不支持对异步操作的取消和没有提供对进度报告的功能,对于有界面的应用程序来说...
    99+
    2022-11-13
  • .NET4.0版本中基于任务的异步模式(TAP)
    一、引言 当使用APM的时候,首先我们要先定义用来包装回调方法的委托,这样难免有点繁琐, 然而使用EAP的时候,我们又需要实现Completed事件和Progress事件,上面两种实...
    99+
    2022-11-13
  • .NET异步编程模式的三种类型介绍
    一、引言 .NET中很多的类、接口在设计的时候都考虑了多线程问题,简化了多线程程序的开发,不用自己去写WaitHandler等这些底层的代码,由于历史的发展,这些类的接口设计有着三种...
    99+
    2022-11-13
  • 基于python yield机制的异步操作同步化编程模型
    本文总结下如何在编写python代码时对异步操作进行同步化模拟,从而提高代码的可读性和可扩展性。 游戏引擎一般都采用分布式框架,通过一定的策略来均衡服务器集群的资源负载,从而保证服务器运算的高并发...
    99+
    2022-06-04
    模型 机制 操作
  • C#中如何使用异步编程模型处理UI响应
    C#中如何使用异步编程模型处理UI响应,需要具体代码示例随着计算机技术的不断发展,用户对于软件系统的响应速度要求也越来越高。传统的同步编程模型在处理复杂的业务逻辑时,容易造成用户界面的卡顿或者无响应问题。为了解决这个问题,C#引入了异步编程...
    99+
    2023-10-22
    C# 异步编程 UI响应
  • C#中如何使用异步任务和并发编程模型
    C#中如何使用异步任务和并发编程模型,需要具体代码示例在C#编程语言中,异步任务和并发编程模型是非常重要的概念和技巧。它们可以帮助我们更好地利用计算资源,提高程序的性能和响应能力。本文将介绍C#中如何使用异步任务和并发编程模型,并提供具体的...
    99+
    2023-10-22
    C# 并发编程 异步任务
  • C#中如何使用并发编程模型处理异步任务
    C#中如何使用并发编程模型处理异步任务,需要具体代码示例引言:在日常的软件开发中,处理异步任务是非常常见的需求。在C#中,我们可以使用并发编程模型来处理异步任务,提高程序的性能和响应能力。本文将介绍C#中的并发编程模型以及如何使用它来处理异...
    99+
    2023-10-22
    异步任务处理 并发编程模型 C#语言
  • C#中如何使用异步编程模型提高响应性能
    C#中如何使用异步编程模型提高响应性能,需要具体代码示例随着计算机性能的提高和互联网的发展,对于高效响应的需求越来越多。在C#中,异步编程模型(Asynchronous Programming Model,简称APM)是一种提高系统响应性能...
    99+
    2023-10-22
    C# 异步编程 异步模型
  • JavaScript中实现异步编程模式的方法
    小编给大家分享一下JavaScript中实现异步编程模式的方法,希望大家阅读完这篇文章之后都有所收获,下面让我们一起去探讨吧!JavaScript中实现异步编程模式的方法:1、回调函数,这是异步编程最基本的方法;2、事件监听;3、发布或订阅...
    99+
    2023-06-14
  • 数据类型的异步编程:Java中的最新趋势?
    随着互联网应用的不断发展,数据处理和数据存储已经成为了最为重要的技术之一。而在Java编程语言中,异步编程已经成为了最新的趋势。异步编程能够提高程序的性能和响应速度,同时也能够减少资源的消耗。 Java中的异步编程主要采用了Future和...
    99+
    2023-08-19
    存储 异步编程 数据类型
  • 数据类型和异步编程:Java中响应式编程的最佳实践
    Java是一门广泛应用于企业级应用程序开发的编程语言。Java语言的强类型特性对于数据类型的定义和处理有着很好的支持。而异步编程技术则可以更好地处理大规模数据和复杂业务逻辑。本篇文章将介绍数据类型和异步编程在Java中的最佳实践,以帮助Ja...
    99+
    2023-10-21
    数据类型 响应 异步编程
  • 如何在 JavaScript 中实现高效的异步编程模式?
    JavaScript 是一门非常流行的编程语言,它的异步编程模式是其最大的特点之一。JavaScript 中的异步编程模式可以帮助我们在处理 I/O 操作和网络请求时提高程序的性能。本文将介绍如何在 JavaScript 中实现高效的异步编...
    99+
    2023-10-02
    异步编程 javascript linux
  • 如何在ASP应用程序中使用异步编程模型来处理并发请求?
    在ASP应用程序中处理并发请求是一个很常见的场景,但是当并发请求量增加时,应用程序可能会变得非常缓慢,甚至崩溃。这时,异步编程模型就成为了解决这个问题的一种方法。 异步编程模型是一种在应用程序中处理并发请求的方法。它可以让应用程序在处理一个...
    99+
    2023-08-04
    存储 异步编程 npm
  • Java中异步编程如何影响数据类型的存储?
    Java是一种面向对象的编程语言,其支持异步编程。在异步编程中,操作在后台线程中执行,而不是在主线程中执行。由于异步编程的特性,它对数据类型的存储有着重要的影响。 Java中异步编程的实现方式有多种,其中最常见的是使用线程池和Future。...
    99+
    2023-08-19
    存储 异步编程 数据类型
  • Python中的异步编程如何处理不同的数据类型?
    随着互联网的发展,越来越多的应用程序需要同时处理多个请求和数据。在传统的同步编程模型下,这些程序往往会受到性能瓶颈的限制,导致响应速度变慢。为了解决这个问题,异步编程模型应运而生。Python作为一门非常流行的编程语言,也提供了很多异步编...
    99+
    2023-08-15
    数据类型 异步编程 二维码
  • C#中如何使用异步编程模型和并发编程处理任务分发及解决方法
    C#中如何使用异步编程模型和并发编程处理任务分发及解决方法引言:在现代的软件开发中,我们经常面临处理大量任务的情况,而这些任务可能是独立的,互不干扰的。为了提高程序的性能和效率,我们希望能够并发地处理这些任务,并且在每个任务完成时能够得到相...
    99+
    2023-10-22
    并发编程 异步编程 任务分发
  • GO语言的异步编程模型:如何优雅地处理文件和日志?
    随着互联网的发展,异步编程模型在软件开发中变得越来越常见。GO语言作为一种高效的编程语言,也具备了异步编程的能力。在这篇文章中,我们将探讨GO语言的异步编程模型,并展示如何使用它来优雅地处理文件和日志。 一、GO语言的异步编程模型 在传统...
    99+
    2023-09-27
    异步编程 文件 日志
  • C#中如何使用异步编程模型和并发集合处理并发问题
    C#中如何使用异步编程模型和并发集合处理并发问题引言:在现代软件开发中,处理并发问题是一个很常见的需求。并发问题指的是多个线程或任务同时访问共享资源,可能导致数据竞争、线程冲突等问题。C#语言提供了异步编程模型和并发集合等工具来帮助我们处理...
    99+
    2023-10-22
    并发集合 异步编程模型 处理并发问题
  • ASP异步编程和JavaScript中的数据类型,哪个更重要?
    随着互联网的快速发展,前端开发和后端开发也越来越受到重视。在这两个方面,ASP异步编程和JavaScript中的数据类型都是非常重要的知识点。但是,哪个更重要呢?这个问题并不好回答,因为它们都有各自的重要性和优缺点。本文将探讨ASP异步编...
    99+
    2023-11-02
    异步编程 javascript 数据类型
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作