5

我有一些文件需要在应用启动(首次运行)时下载。我在 Windows 8 中使用后台下载器。这就是我使用它的方式:

BackgroundDownloader downloader = new BackgroundDownloader();

List<DownloadOperation> operations = new List<DownloadOperation>();
foreach (FileInfo info in infoFiles)
{
    Windows.Storage.ApplicationData.Current.LocalFolder;
    foreach (string folder in info.Folders)
    {
        currentFolder = await currentFolder.CreateFolderAsync(folder, CreationCollisionOption.OpenIfExists);       
    }
    //folder hierarchy created, save the file
    StorageFile file = await currentFolder.CreateFileAsync(info.FileName, CreationCollisionOption.ReplaceExisting);

    DownloadOperation op = downloader.CreateDownload(new Uri(info.Url), file);
    activeDownloads.Add(op);
    operations.Add(op);
}


foreach (DownloadOperation download in operations)
{
  //start downloads
  await HandleDownloadAsync(download, true);
}

当我尝试使用时,BackgroundDownloader.GetCurrentDownloadsAsync();我只发现了一个后台下载。所以我删除了上面的 await 操作符,让它们异步启动。

但是我需要知道所有文件何时完成,以便设置进度条。

我需要一种下载多个文件、发现所有文件BackgroundDownloader.GetCurrentDownloadsAsync();并知道所有下载何时完成的方法。

  private async Task HandleDownloadAsync(DownloadOperation download, bool start)
    {
        try
        {
            Debug.WriteLine("Running: " + download.Guid, NotifyType.StatusMessage);

            // Store the download so we can pause/resume.
            activeDownloads.Add(download);

            Progress<DownloadOperation> progressCallback = new Progress<DownloadOperation>(DownloadProgress);
            if (start)
            {
                // Start the download and attach a progress handler.
                await download.StartAsync().AsTask(cts.Token, progressCallback);
            }
            else
            {
                // The download was already running when the application started, re-attach the progress handler.
                await download.AttachAsync().AsTask(cts.Token, progressCallback);
            }

            ResponseInformation response = download.GetResponseInformation();

            Debug.WriteLine(String.Format("Completed: {0}, Status Code: {1}", download.Guid, response.StatusCode),
                NotifyType.StatusMessage);
        }
        catch (TaskCanceledException)
        {
            Debug.WriteLine("Canceled: " + download.Guid, NotifyType.StatusMessage);
        }
        catch (Exception ex)
        {
            if (!IsExceptionHandled("Execution error", ex, download))
            {
                throw;
            }
        }
        finally
        {
            activeDownloads.Remove(download);
        }
    }
4

1 回答 1

3

如果您特别需要在每次应用程序启动时重新下载文件,则不应使用 BackgroundDownloader - 它适用于即使您的应用程序暂停然后重新启动时仍需要继续下载的情况。如果您只是要在下次再次启动应用程序时再次重新下载文件,那么一旦您的应用程序消失,您就允许继续下载,这只是在浪费电池。

在 Windows 应用商店应用程序中,您有几个不支持后台下载的选项。最直接的方法是 Windows.Web.Http.HttpClient -这是一个很好的入门指南,介绍如何使用它。但是,它仅适用于 Windows 8.1 应用程序,不适用于 Windows 8.0。如果您需要 Windows 8.0 兼容性,那么独特但名称相似的 System.Net.Http.HttpClient 可能是您的最佳选择。Harrison 已经提供了相关快速入门文档的链接。

如果您确实需要后台下载功能,请修改您的问题以更准确地反映您的情况。当然可以使用 BackgroundDownloader 同时启动多个后台下载并将进度和完成处理程序附加到每个下载,但这听起来不是您问题场景的最佳解决方案。


既然您已经阐明了用法,那么问题就更加清楚了。您的HandleDownloadAsync方法是将要下载的进度处理程序附加到下载中,但随后它正在等待下载完成。当你写

foreach (DownloadOperation download in operations)
{
    //start downloads
    await HandleDownloadAsync(download, true);
}

因此,在开始下一次下载之前,您依次等待每次下载完成。

此处的解决方案是执行所有调用之前不要等待调用结果- 这将立即启动每个下载并附加适当的进度处理程序,而无需等待其中任何一个完成。修改后的版本可能如下所示:HandleDownloadAsync

BackgroundDownloader downloader = new BackgroundDownloader();

List<Task> downloadCompletionTasks = new List<Task>();
foreach (FileInfo info in infoFiles)
{
    StorageFile file = /* ... */;

    DownloadOperation op = downloader.CreateDownload(new Uri(info.Url), file);

    // Consider moving this line into HandleDownloadAsync so you won't have to repeat
    // it in the code that's handling the GetCurrentDownloadsAsync logic.
    activeDownloads.Add(op);

    // Starting the download, but not awaiting its completion yet
    Task downloadCompletionTask = HandleDownloadAsync(op, true);
    downloadCompletionTasks.Add(downloadCompletionTask);
}

// Now that we've got all the downloads started concurrently, we can wait until all of them complete
await Task.WhenAll(downloadCompletionTasks);
于 2013-10-30T01:26:00.963 回答