0

我正在尝试编写一个 C# Windows 应用程序来下载文件(等待下载完成 - 提供一些进度指示器),然后执行下载的文件。我想我要么需要:

  • 具有进度指示器且不会使我的应用程序无响应的同步下载
  • 等待下载完成然后尝试执行它的异步下载。

前者(同步下载)似乎可以满足我的要求,但我找不到任何指示进度的方法,而且它似乎使我的程序无响应(就像它挂起一样)——这可能会导致用户关闭应用程序等待它完成下载。

后来(异步下载)我已经能够使用进度指示器等来完成,但发生的是下载开始并且我的应用程序在完成下载之前立即尝试安装它,这当然会失败。

那么实现这一目标的最佳方法是什么?下面是我目前使用进度条进行异步下载的代码。

编辑 2012 年 4 月 10 日

旧代码变得很麻烦,所以这就是我目前拥有的。它可以工作,只是它不更新进度条。

    private void backgroundWorker1_DoWork(object sender, System.ComponentModel.DoWorkEventArgs e)
    {

            List<App> FilesToDownload = e.Argument as List<App>;

            foreach (App Program in FilesToDownload) // Loop through List with foreach
            {
                //string Text = ;
                UpdateRichTextBox("Downloading " + Program.Filename + "... Please wait");

                string Source = Program.DownloadLocation + "/" +  Program.Filename;
                string Destination = System.AppDomain.CurrentDomain.BaseDirectory + Program.Filename;

                WebClient webClient = new WebClient();
                webClient.DownloadFile(Source, @Destination);
            }

            UpdateRichTextBox("File download complete");

            foreach (App Program in FilesToDownload) // Loop through List with foreach
            {
                string Filename = Program.Filename;
                string Arguments = Program.InstallParameters;

                if (Filename != "advisorinstaller.exe")
                {
                    Process p = new Process();
                    p.StartInfo.FileName = System.AppDomain.CurrentDomain.BaseDirectory + Filename;
                    p.StartInfo.Arguments = Arguments;
                    p.Start();
                    p.WaitForExit();
                }
                else
                {
                    MessageBox.Show("About to install Belarc Advisor. This may take 10-15 minutes to run - do not shutdown this program. Click OK when ready to proceed.");
                }
            }
    }

    private void backgroundWorker1_ProgressChanged(object sender, System.ComponentModel.ProgressChangedEventArgs e)
    {
        progressBar1.Value = e.ProgressPercentage;
    }

    private void backgroundWorker1_RunWorkerCompleted(object sender, System.ComponentModel.RunWorkerCompletedEventArgs e)
    {
        UpdateRichTextBox("Install Complete");

    }

编辑 2012 年 4 月 11 日

所以我编辑了我的后台工作人员的工作如下:

        private void backgroundWorker1_DoWork(object sender, System.ComponentModel.DoWorkEventArgs e)
        {

            List<App> FilesToDownload = e.Argument as List<App>;

            int counter = 0;
            int percent = 0;

            foreach (App Program in FilesToDownload) // Loop through List with foreach
            {
                percent = ((counter / FilesToDownload.Count) * 100);
                backgroundWorker1.ReportProgress(percent, "");

                UpdateRichTextBox("Downloading " + Program.Filename + "... Please wait");

                string Source = Program.DownloadLocation + "/" +  Program.Filename;
                string Destination = System.AppDomain.CurrentDomain.BaseDirectory + Program.Filename;

                WebClient webClient = new WebClient();
                webClient.DownloadFile(Source, @Destination);


                counter++;

            }

            //backgroundWorker1.ReportProgress(100, "Complete!");
    }

如果我取消注释最后一行,进度条会变为 100%。但它没有使用:

percent = ((counter / FilesToDownload.Count) * 100);
backgroundWorker1.ReportProgress(percent, "");

任何想法为什么?

4

4 回答 4

1

您可以使用 aBackgroundWorker在后台执行同步下载。它也支持进度报告。这将避免阻塞您的 UI(因为下载在不同的线程上运行)。

于 2012-04-04T18:57:49.410 回答
0

您可以使用ManualResetEvent类在主线程上等待(使用 WaitOne()),直到调用您的 client_DownloadFileCompleted() 事件处理程序(在您的对象上调用 Set())。

于 2012-04-04T19:01:16.740 回答
0

无需将其包装WebClient在 aBackgroundWorker中,因为它已经是异步的(使用线程池中的线程)。

注意:下面的示例是在编辑器中编写的,而我检查了 MSDN 文档中的方法签名,我很容易出错。

public void InstallSecuniaPSI()
{
    var source = new Uri("http://www.webserver.com:8014/psisetup.exe");
    var dest = Path.Combine(Path.GetTemp(), "Download_SecuniaPSI.exe");

    var client = new WebClient();
    client.DownloadProgressChanged += OnDownloadProgressChanged;
    client.DownloadFileCompleted += OnDownloadComplete;
    client.DownloadFileAsync(source, dest, dest);   
}

private void OnDownloadProgressChanged(object sender, DownloadProgressChangedEventArgs e)
{
    // report progress
    Console.WriteLine("'{0}' downloaded {1} of {2} bytes. {3}% complete"(string)e.UserState, e.BytesReceived, e.TotalBytesToReceive, e.ProgressPercentage);
}

private void OnDownloadComplete(object sender, AsyncCompletedEventArgs e)
{
    // clean up
    var client = (WebClient)sender;
    client.DownloadProgressChanged -= OnDownloadProgressChanged;
    client.DownloadFileCompleted -= OnDownloadComplete;

    if (e.Error != null)
        throw e.Error;

    if (e.Cancelled)
        Environment.Exit(1);

    // install program
    var downloadedFile = (string)e.UserState;
    var processInfo = new ProcessStartInfo(downloadedFile, "/S");
    processInfo.CreateNoWindow = true;

    var installProcess = Process.Start(processInfo);
    installProcess.WaitForExit();

    Environment.Exit(installProcess.ExitCode);
}

现在唯一需要整理的是如何让程序等待;如果是控制台程序,则Console.ReadKey()InstallSecuniaPSI().

高温下,

于 2012-04-04T22:37:41.833 回答
0

您是否考虑过使用ClickOnce来执行此操作?它将与自动更新一起解决您的问题。

查看这个 SO 问题以了解开始的地方:https ://stackoverflow.com/questions/1635103/how-to-basic-tutorial-to-write-a-clickonce-app 。

于 2012-04-04T19:03:28.517 回答