C# 异步下载文件

简介: 在C#当中,利用WebClient这个核心类,可以轻易的打造一个下载器。但是这里想要强调的是,我们用的是异步操作。

在C#当中,利用WebClient这个核心类,可以轻易的打造一个下载器。但是这里想要强调的是,我们用的是异步操作。所谓异步,是相对于同步的概念而言的。比如Web中的Ajax就是基于异步的。它能够提供良好的用户体验,让用户在进行操作时,不感觉到“卡”(不阻塞UI线程),能够同时进行其它的操作并能够随意的切换到任务界面。在下载文件时,如果文件过大,我们用同步的下载方式进行下载会感觉程序“假死”,其实程序在后台不断的运行,但我们看不到下载的过程。所以这时候使用异步方法能够有效的解决这个问题。

先看一下程序的界面:


实现上面的操作很简单,只需要几行代码就可以搞定。

        private void button1_Click(object sender, EventArgs e)
        {
            using (WebClient client = new WebClient())
            {
                client.DownloadFileAsync(new Uri(this.textBox1.Text.Trim()),Path.GetFileName(this.textBox1.Text.Trim()));
                client.DownloadProgressChanged += client_DownloadProgressChanged;
                client.DownloadFileCompleted += client_DownloadFileCompleted;
            }
        }

        void client_DownloadProgressChanged(object sender, DownloadProgressChangedEventArgs e)
        {
            this.label1.Text = string.Format("当前接收到{0}字节,文件大小总共{1}字节", e.BytesReceived, e.TotalBytesToReceive);
            this.progressBar1.Value = e.ProgressPercentage;
        }

        void client_DownloadFileCompleted(object sender, System.ComponentModel.AsyncCompletedEventArgs e)
        {
            if (e.Cancelled)
            {
                MessageBox.Show("文件下载被取消", "提示", MessageBoxButtons.OKCancel);
            }
            this.progressBar1.Value = 0;
            MessageBox.Show("文件下载成功", "提示");
        }
我们只需要在textbox中填入文件的地址,比如迅雷的下载地址:http://dlsw.baidu.com/sw-search-sp/soft/ca/13442/Thunder_dl_7.9.18.4706Preview.2205245239.exe,就可以用上面的代码进行下载了。

在C#当中,还可以利用HttpWebRequest进行文件的异步下载。下面的代码可能稍微有点复杂,但是可以帮助我们深入理解“异步“操作的过程。

我们先定义一个类,用于保存操作的状态:

    /// <summary>
    /// 请求状态
    /// </summary>
    public class RequestState
    {
        /// <summary>
        /// 缓冲区大小
        /// </summary>
        public int BUFFER_SIZE { get; set; }

        /// <summary>
        /// 缓冲区
        /// </summary>
        public byte[] BufferRead { get; set; }

        /// <summary>
        /// 保存路径
        /// </summary>
        public string SavePath { get; set; }

        /// <summary>
        /// 请求流
        /// </summary>
        public HttpWebRequest Request { get; set; }

        /// <summary>
        /// 响应流
        /// </summary>
        public HttpWebResponse Response { get; set; }

        /// <summary>
        /// 流对象
        /// </summary>
        public Stream ResponseStream { get; set; }

        /// <summary>
        /// 文件流
        /// </summary>
        public FileStream FileStream { get; set; }
    }
在一个Button的Click事件下,键入如下代码:

            //下载文件的url
            string url = this.textBox1.Text.Trim();

            //创建一个初始化请求对象
            HttpWebRequest request = (HttpWebRequest)WebRequest.Create(new Uri(url));

            //设置下载相关参数
            RequestState requestState = new RequestState();
            requestState.BUFFER_SIZE = 1024;
            requestState.BufferRead = new byte[requestState.BUFFER_SIZE];
            requestState.Request = request;
            requestState.SavePath = Path.Combine("D:\\", Path.GetFileName(url));
            requestState.FileStream = new FileStream(requestState.SavePath, FileMode.OpenOrCreate);

            //开始异步请求资源
            request.BeginGetResponse(new AsyncCallback(ResponseCallback), requestState);
我们可以看到,异步的操作方法一般都是以Begin开头的BeginGetResponse,我们平时用的比较多的同步方法直接使用GetResponse。另外AsyncCallback是一个委托,前面讲过,它里面的参数是一个方法,我们起名为ResponseCallback,并且把requestState作为参数传递过去。

接下来就可以看一下ResponseCallback方法:

        /// <summary>
        /// 请求资源方法的回调函数
        /// </summary>
        /// <param name="asyncResult">用于在回调函数当中传递操作状态</param>
        private void ResponseCallback(IAsyncResult asyncResult)
        {
            RequestState requestState = (RequestState)asyncResult.AsyncState;
            requestState.Response = (HttpWebResponse)requestState.Request.EndGetResponse(asyncResult);

            Stream responseStream = requestState.Response.GetResponseStream();
            requestState.ResponseStream = responseStream;

            //开始异步读取流
            responseStream.BeginRead(requestState.BufferRead, 0, requestState.BufferRead.Length, ReadCallback, requestState);
        }
我们可以看到,回调函数里面又有一个异步操作。它的任务是对响应流异步的读取到缓冲区当中。

再进一步,看一下ReadCallback回调函数。

        /// <summary>
        /// 异步读取流的回调函数
        /// </summary>
        /// <param name="asyncResult">用于在回调函数当中传递操作状态</param>
        private void ReadCallback(IAsyncResult asyncResult)
        {
            RequestState requestState = (RequestState)asyncResult.AsyncState;
            int read = requestState.ResponseStream.EndRead(asyncResult);
            if (read > 0)
            {
                //将缓冲区的数据写入该文件流
                requestState.FileStream.Write(requestState.BufferRead, 0, read);

                //开始异步读取流
                requestState.ResponseStream.BeginRead(requestState.BufferRead, 0, requestState.BufferRead.Length, ReadCallback, requestState);
            }
            else
            {
                requestState.Response.Close();
                requestState.FileStream.Close();
            }
        }

这里面是真正的将流写入文件的过程,并且用BeginRead方法递归的写入文件流直到文件完全写好为止(完全下载到本地)。

上面我参考了官方网站上面的代码,可以在这里查询到:BeginGetResponse。这是一个经典的异步操作的例子,希望大家能够好好理解。

目录
相关文章
|
人工智能 前端开发 机器人
虚拟数字人开放平台产品分享
本文摘自虚拟数字人新品发布会姜望讲解部分
2086 0
虚拟数字人开放平台产品分享
|
存储 SQL 分布式计算
数据湖 VS 数据仓库之争?阿里提出大数据架构新概念:湖仓一体
随着近几年数据湖概念的兴起,业界对于数据仓库和数据湖的对比甚至争论就一直不断。有人说数据湖是下一代大数据平台,各大云厂商也在纷纷的提出自己的数据湖解决方案,一些云数仓产品也增加了和数据湖联动的特性。但是数据仓库和数据湖的区别到底是什么,是技术路线之争?是数据管理方式之争?二者是水火不容还是其实可以和谐共存,甚至互为补充?本文作者来自阿里巴巴计算平台部门,深度参与阿里巴巴大数据/数据中台领域建设,将从历史的角度对数据湖和数据仓库的来龙去脉进行深入剖析,来阐述两者融合演进的新方向——湖仓一体,并就基于阿里云MaxCompute/EMR DataLake的湖仓一体方案做一介绍。
28994 2
数据湖 VS 数据仓库之争?阿里提出大数据架构新概念:湖仓一体
|
区块链 Android开发
区块链大众化的落地产品-深度体验蚂蚁区块链鹊凿数字版权服务平台
从18年至今,一直在思考区块链的真正有价值的大众落地是什么。 18年在个人有限的认知里得出的结论是: 围绕着空气币是没有太多的价值,只对于黑灰色产业有价值。 19年末关注到了,蚂蚁链的-鹊凿数字版权版权平台。 时至今日,经历一年半的时光,2021年,最近看到了淘宝商家服务的成交量激增,和蚂蚁与杭州互联网公证处的深度合作。确信在未来3-5年,这是一个很好的历史市场环境。
2001 1
区块链大众化的落地产品-深度体验蚂蚁区块链鹊凿数字版权服务平台
|
Java 应用服务中间件
SpringBoot集成使用jsp(超详细)
SpringBoot集成使用jsp(超详细)
SpringBoot集成使用jsp(超详细)
|
编解码 前端开发 算法
基于OpenCV的双目摄像头测距(误差小)
首先进行双目摄像头定标,获取双目摄像头内部的参数后,进行测距;本文的双目视觉测距是基于BM算法。注意:双目定标的效果会影响测距的精准度,建议大家在做双目定标时,做好一些(尽量让误差小)。
12235 3
基于OpenCV的双目摄像头测距(误差小)
|
8月前
|
人工智能 缓存 自然语言处理
保姆级Spring AI 注解式开发教程,你肯定想不到还能这么玩!
这是一份详尽的 Spring AI 注解式开发教程,涵盖从环境配置到高级功能的全流程。Spring AI 是 Spring 框架中的一个模块,支持 NLP、CV 等 AI 任务。通过注解(如自定义 `@AiPrompt`)与 AOP 切面技术,简化了 AI 服务集成,实现业务逻辑与 AI 基础设施解耦。教程包含创建项目、配置文件、流式响应处理、缓存优化及多任务并行执行等内容,助你快速构建高效、可维护的 AI 应用。
|
机器学习/深度学习 并行计算 数据可视化
目标分类笔记(二): 利用PaddleClas的框架来完成多标签分类任务(从数据准备到训练测试部署的完整流程)
这篇文章介绍了如何使用PaddleClas框架完成多标签分类任务,包括数据准备、环境搭建、模型训练、预测、评估等完整流程。
974 0
目标分类笔记(二): 利用PaddleClas的框架来完成多标签分类任务(从数据准备到训练测试部署的完整流程)
|
小程序 前端开发
【非常全】微信小程序下载图片到相册,微信小程序自动获取分享图片到相册
【非常全】微信小程序下载图片到相册,微信小程序自动获取分享图片到相册
1204 3
|
小程序 开发工具 开发者
【微信小程序】微信开发者工具 引用 vant-weapp时“miniprogram/node_modules/@babel/runtime/index.js: 未找到npm包入口文件” 解决办法
【微信小程序】微信开发者工具 引用 vant-weapp时“miniprogram/node_modules/@babel/runtime/index.js: 未找到npm包入口文件” 解决办法
1225 1
|
安全 网络协议 Go
gRPC- HTTP网关 I
gRPC- HTTP网关 I
329 0