如何使用progressBar为程序中的每个进程执行进度?

时间:2013-07-30 21:58:36

标签: c# winforms progress-bar

在表格的顶部我做了:

progressBar1.Maximum = 100;
progressBar1.Minimum = 1;

然后在按钮点击事件开始我做的操作:

timer2.Enabled = true;
if (this.backgroundWorker1.IsBusy == false)
            {
                this.backgroundWorker1.RunWorkerAsync();
            }

然后在backgroundworkerdowork事件中:

private void backgroundWorker1_DoWork(object sender, DoWorkEventArgs e)
        {
            BackgroundWorker worker = sender as BackgroundWorker;
            if (filesContent.Length > 0)
            {
                for (int i = 0; i < filesContent.Length; i++)
                {
                    File.Copy(filesContent[i], Path.Combine(contentDirectory, Path.GetFileName(filesContent[i])), true);
                }
            }
            DoProgressBar(e, worker);
            WindowsUpdate();
            CreateDriversList();
            GetHostsFile();
            Processes();
        }

DoWork事件中的所有函数都是复制文件,Processes()函数使用我使用Process创建/复制文件的新类。

然后我做了新的DoProgressBar事件函数:

private static void DoProgressBar(DoWorkEventArgs e, BackgroundWorker worker)
        {
            for (int i = 1; i <= 90; i++)
            {
                if ((worker.CancellationPending == true))
                {
                    e.Cancel = true;
                    break;
                }
                else
                {
                    // Perform a time consuming operation and report progress.
                    System.Threading.Thread.Sleep(50);
                    worker.ReportProgress(i);
                }
            }
        }

然后是ProgressChanged事件:

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

然后完成的事件:

private void backgroundWorker1_RunWorkerCompleted(object sender, RunWorkerCompletedEventArgs e)
        {
            if ((e.Cancelled == true))
            {
                this.label1.Text = "Canceled!";
            }
            else if (!(e.Error == null))
            {
                this.label1.Text = ("Error: " + e.Error.Message);
            }
            else
            {
                this.progressBar1.Value = this.progressBar1.Maximum;
                processfinish = true;
            }
        }

Timer2 tick事件:

private void timer2_Tick(object sender, EventArgs e)
        {
            timerCount += 1;
            TimerCount.Text = TimeSpan.FromSeconds(timerCount).ToString();
            TimerCount.Visible = true;
            if (processfinish == true)
            {
                timer2.Enabled = false;
                timer1.Enabled = true;
            }                           
        }

并且timer1 tick事件:

private void timer1_Tick(object sender, EventArgs e)
        {
            count++;
            Diagnose.Text = "PROCESS HAS FINISHED" + "  " + countBack--;
            if (count == 6)
            {
                Diagnose.Text = "COLLECT INFORMATION";
                Diagnose.Enabled = true;
                CreateZip.Enabled = true;
                ViewLogFile.Enabled = true;
                DriverVerifier.Enabled = true;
                timer1.Enabled = false;
                TimerCount.Visible = false;
            }
        }

我知道它的代码很长,但这里的所有内容都已连接。

我想要做的是progressBar将根据DoWork事件中每个函数的进度获得进展。

但现在它正在做的是首先进入:

DoProgressBar()事件/函数执行第二个/ else部分ReportProgress(i)

然后进入Progresschanged事件并执行:progressBar1.Value = e.ProgressPercentage;

结果是当我点击按钮点击开始操作时,我立即看到进度条几乎移动到最后,而不是根据程序的每个功能/进度移动。

您可以在此处查看我的完整Form1代码:

http://codepaste.net/fuk9w5

编辑:

这是ProcessRun类的代码,其中im在函数Processes()中的Form1中使用

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Diagnostics;
using System.IO;

namespace Diagnostic_Tool_Blue_Screen
{
    class ProcessRun
    {


        public void ProcessesRun()
        {

        }

        public static void Processing(string WorkingDirectory, string FileName, string Arguments, bool StandardOutput, string OutputFileName)
        {
            Process proc = new Process();
            proc.EnableRaisingEvents = true;
            proc.StartInfo.UseShellExecute = false;
            proc.StartInfo.RedirectStandardOutput = StandardOutput;
            proc.StartInfo.FileName = FileName;
            proc.StartInfo.CreateNoWindow = true;
            proc.StartInfo.WorkingDirectory = WorkingDirectory;
            proc.StartInfo.Arguments = Arguments;
            proc.Start();
            if (StandardOutput == true)
            {
                string output = proc.StandardOutput.ReadToEnd();
                DumpOutput(WorkingDirectory + "\\" + OutputFileName, output);
            }
            proc.WaitForExit();
            proc.Close();
        }

        private static void DumpOutput(string filename, string output)
        {
            StreamWriter w = new StreamWriter(filename);
            w.Write(output);
            w.Close();
        }
    }
}

3 个答案:

答案 0 :(得分:0)

您的后台线程似乎直接与UI元素(进度条)进行交互。那是个问题。您的后台线程无法直接与UI元素交互;它必须调用它,以便UI更新发生在UI线程上。

例如,您可以在表单中添加这样的方法:

// Form method for updating progress bar; callable from worker thread
public void UpdateProgressBar(double progress)
{
    // dispatch the update onto the form's thread
    Dispatcher.BeginInvoke((Action<double>)((n) =>
    {
        // do the update in the form's thread
        progressBar1.Value = n;
    }), progress);
}

然后,您可以从工作线程调用此方法,并且进度条应该正确更新。

答案 1 :(得分:0)

为什么不把它放到你的代码中?这就是我移动进度条的“绿色”部分的方式。

progressBar1.Step = pos; //where pos is the number on how much do you want to increase the progress of the progressbar
progressBar1.PerformStep(); //triggers the movement of the progressBar.

答案 2 :(得分:0)

对于那些在试图找到一种方法来改变progess bar的交叉线程的人来到这里时,你就是这样做的:

form.Invoke((Action)delegate { form.function(); });