带进度条的 azure blob 存储异步下载

Pau*_*ien 5 c# azure azure-storage-blobs

我正在尝试使用连接到进度条的 .DownloadToStreamAsync() 方法从 Azure Blob 存储下载文件的完整示例。

我发现了对 azure 存储 sdk 旧实现的引用,但它们不能与较新的 sdk(已实现这些异步方法)一起编译,或者不能与当前的 nuget 包一起使用。

https://blogs.msdn.microsoft.com/avkashchauhan/2010/11/03/uploading-a-blob-to-azure-storage-with-progress-bar-and-variable-upload-block-size/

https://blogs.msdn.microsoft.com/kwill/2013/03/05/asynchronous-parallel-blob-transfers-with-progress-change-notification-2-0/

我是 .NET 中异步/等待线程的新手,想知道是否有人可以帮助我解决以下问题(在 Windows 窗体应用程序中)并展示我如何“挂钩”文件下载的进度。 .. 我看到一些示例不使用 .DownloadToStream 方法,而是下载 blob 文件的块。可以做吗?

因此,假设以下工作正常(非异步),我还需要做什么才能使用 blockBlob.DownloadToStream Async (fileStream); 方法,这甚至是正确的使用方法,我怎样才能取得进展?

理想情况下,我以任何方式都可以挂钩 blob 下载的进度,以便我可以在大下载时更新 Windows 窗体 UI..所以如果下面的方法不正确,请赐教:)

// Retrieve storage account from connection string.
CloudStorageAccount storageAccount = CloudStorageAccount.Parse(
    CloudConfigurationManager.GetSetting("StorageConnectionString"));

// Create the blob client.
CloudBlobClient blobClient = storageAccount.CreateCloudBlobClient();

// Retrieve reference to a previously created container.
CloudBlobContainer container = blobClient.GetContainerReference("mycontainer");

// Retrieve reference to a blob named "photo1.jpg".
CloudBlockBlob blockBlob = container.GetBlockBlobReference("photo1.jpg");

// Save blob contents to a file.
using (var fileStream = System.IO.File.OpenWrite(@"path\myfile"))
{
    blockBlob.DownloadToStream(fileStream);
}
Run Code Online (Sandbox Code Playgroud)

使用Gaurav建议的很棒的建议方法(下载 1mb 块),我已经使用后台工作人员来实现下载,这样我就可以随时更新 UI。

do 循环中的主要部分将范围下载到流,然后将流写入我没有从原始示例中触及的文件系统,但我添加了代码来更新工作进程并侦听工作取消(以中止下载).. 不确定这是否可能是问题?

为了完整起见,以下是 worker_DoWork 方法中的所有内容:

public void worker_DoWork(object sender, DoWorkEventArgs e)
    {
        object[] parameters = e.Argument as object[];
        string localFile = (string)parameters[0];
        string blobName = (string)parameters[1];
        string blobContainerName = (string)parameters[2];
        CloudBlobClient client = (CloudBlobClient)parameters[3];      

        try
        {
            int segmentSize = 1 * 1024 * 1024; //1 MB chunk
            var blobContainer = client.GetContainerReference(blobContainerName);
            var blob = blobContainer.GetBlockBlobReference(blobName);
            blob.FetchAttributes();
            blobLengthRemaining = blob.Properties.Length;
            blobLength = blob.Properties.Length;
            long startPosition = 0;
            do
            {
                long blockSize = Math.Min(segmentSize, blobLengthRemaining);
                byte[] blobContents = new byte[blockSize];
                using (MemoryStream ms = new MemoryStream())
                {
                    blob.DownloadRangeToStream(ms, startPosition, blockSize);
                    ms.Position = 0;
                    ms.Read(blobContents, 0, blobContents.Length);
                    using (FileStream fs = new FileStream(localFile, FileMode.OpenOrCreate))
                    {
                        fs.Position = startPosition;
                        fs.Write(blobContents, 0, blobContents.Length);
                    }
                }
                startPosition += blockSize;
                blobLengthRemaining -= blockSize;

                if (blobLength > 0)
                {
                    decimal totalSize = Convert.ToDecimal(blobLength);
                    decimal downloaded = totalSize - Convert.ToDecimal(blobLengthRemaining);
                    decimal blobPercent = (downloaded / totalSize) * 100;
                    worker.ReportProgress(Convert.ToInt32(blobPercent));
                }

                if (worker.CancellationPending)
                {
                    e.Cancel = true;
                    blobDownloadCancelled = true;
                    return;
                }
            }
            while (blobLengthRemaining > 0);
        }
        catch (Exception ex)
        {
            MessageBox.Show(ex.Message);
        }
    }
Run Code Online (Sandbox Code Playgroud)

这是有效的,但在更大的文件(例如 30mb)上,我有时会收到“无法写入文件,因为在另一个进程错误中打开......”并且进程失败..

Gau*_*tri 1

使用您的代码:

using (var fileStream = System.IO.File.OpenWrite(@"path\myfile"))
{
    blockBlob.DownloadToStream(fileStream);
}
Run Code Online (Sandbox Code Playgroud)

无法显示进度,因为只有当下载完成时代码才会从该函数中出来。DownloadToStream函数将在内部将一个大的 blob 分割成块并下载这些块。

您需要做的是使用您的代码下载这些块。你所要做的就是使用DownloadRangeToStream方法。我之前回答过一个类似的问题,您可能会觉得有用:Azure 下载 blob 部分