在C#中从线程更新进度条

4
    public class Consumer
    {
        Queue<int> queue;
        Object lockObject;

        public Consumer(Queue<int> queue, Object lockObject)
        {
            this.queue = queue;
            this.lockObject = lockObject;
        }

        public void consume(string filepath)
        {
            int item = 0;


            while (true)
            {
                lock (lockObject)
                {
                    if (queue.Count == 0)
                    {
                        Monitor.PulseAll(lockObject);
                        continue;
                    }

                    item = queue.Dequeue();
                    if (item == 0)
                    {
                        break;
                    }

                   //do some staff
                }

            }

        }
    }




    public class Producer 
    {
        Queue<int> queue;
        Object lockObject;

        public int ProgressPercent = 0;
        int TotalProducedElements = 0;
        public bool check1 = false;

        public Producer(Queue<int> queue, Object lockObject)
        {
            this.queue = queue;
            this.lockObject = lockObject;
        }

        private bool IsPrime(int num)
        {
            if (num == 0)
                return true;
            num = Math.Abs(num);
            for (int i = 2; i <= Math.Sqrt(num); i++)
                if (num % i == 0)
                    return false;
            return true;
        }

        public void produce(int target)
        { 
            try
            {
                int seq = 0;
                while (seq++ < target)
                {

                    lock (lockObject)
                    {
                        int item = seq;
                        if (IsPrime(item))
                        {
                            queue.Enqueue(item);

                        }
                        TotalProducedElements++;

                        ProgressPercent = seq;

                        if (queue.Count == 0)
                        {
                            Monitor.PulseAll(lockObject);
                        }
                    }
                }
                queue.Enqueue(0);
              }
            catch (Exception e)
            {
            }

        }
    }
}

 private void Start_Click(object sender, EventArgs e)
    {           

                    Object lockObj = new object(); 

                    Queue<int> queue = new Queue<int>();  

                    Producer p = new Producer(queue, lockObj);

                    Consumer c = new Consumer(queue, lockObj);



                    int target = int.Parse(TargetText.Text);
                    string path = Path.Text;

                    Thread Thread1 = new Thread(() => p.produce(target));
                    Thread Thread2 = new Thread(()=>c.consume(path));

                    Thread1.Start();



                    Thread2.Start();
                   progressBar1.Maximum = target;

                  while(true)
                       {
                         if(p.ProgressPercent==0)
                            {
                               Thread.sleep(1000);
                            }
                            else
                             {
                               progressBar1.Value=p.ProgressPercent;
                             }

                        }

                 }

我有两个类在同一个队列上工作。其中一个类用于生成一组整数,另一个类用于从队列中消耗这些整数。
在此过程中,我希望通过该百分比更新我的进度条。那么如何在不阻塞GUI的情况下从消费者更新进度条?
请注意,我已经使用了 progressbar.Invokedelegate,但没有起作用。

6
你有没有想过将你的例子缩减到绝对必要的最小限度?控制台输出、性能分析、异常处理等对读者来说并不是非常有吸引力,对吗? - Vlad
2
清理一下你的代码,这样你就可以更快地得到帮助。另外,看一下invoke方法http://msdn.microsoft.com/en-us/library/vstudio/zyzhdc6b.aspx,以便从另一个线程更新UI。 - Jonathan Muller
3个回答

10

你需要两件事情,第一是线程,另一个是调用者。

调用者将允许你从线程内部更改窗口控件。以下是示例:

Invoke((MethodInvoker)delegate
{
    Label1.text = "asd";
}

线程的运行方式如下:

some_thread = new Thread
(delegate()
{

    {
        //some_thread code
    }
});
some_thread.Start();

在文件开头添加using System.Threading;

因此,最终代码应该是这样的:

some_thread = new Thread
(delegate()
{

    {
        for(;;)
        {
                Invoke((MethodInvoker)delegate
                {
                                //update the progress bar here
                }
        }
    }
});
some_thread.Start();

我想这值得一试。诀窍在于在更新进度条之前使用Invoker。 - d33tah
(或者,更新Invoker内部) - d33tah
为什么在ASP.NET中Invoke((MethodInvoker)delegate { })不起作用? - user2530748
如果委托代码中没有循环代码(例如for(;;)),而不是使用复杂的算法解决问题。那么我们在哪里调用invoke呢?是在开始、结束还是有选择地放置在算法内部? - Raulp
@Raulp:回复晚了,非常抱歉。当您要触摸UI时需要将其放置。 - d33tah

2

0

清除:

while(true)
                   {
                     if(p.ProgressPercent==0)
                        {
                           Thread.sleep(1000);
                        }
                        else
                         {
                           progressBar1.Value=p.ProgressPercent;
                         }

                    }

替换:

ProgressPercent = seq;

UpdateProgres(seq);

地址:

public void UpdateProgres(int _value)
    {
        if (InvokeRequired)
        {
            Invoke(new Action<int>(UpdateProgres), _value);
            return;
        }

        progressBar1.Value = _value;
    }

总结你所需的函数;

public void UpdateProgres(int _value)
    {
        if (InvokeRequired)
        {
            Invoke(new Action<int>(UpdateProgres), _value);
            return;
        }

        progressBar1.Value = _value;
    }

或者需要锁定该值。例如:

while(true)
{
    lock (p.ProgressPercent)
    {
        int _p = p.ProgressPercent;
    }
    if(_p==0)
    {
        Thread.sleep(1000);
    }
    else
    {
        progressBar1.Value=_p;
    }
}

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接