How to create percentage of loop(processing) [c#]

2.3k Views Asked by At

example : doing something 9999 time (maybe more than)

for (int i = 1; i <= 9999; i++)
{
    // do something
    label1.content = 100*i/9999 + "%" ;
}

and I want to show percentage of loop on label1 when I compile I can't doing anything several a millisecond and my label show 100% only. someone have any idea sir? thank you.

3

There are 3 best solutions below

3
On BEST ANSWER

You can't run a loop on and the update the UI on the same thread simultaneously. That's why you should always perform any long-running work on a background thread and update UI at regular intervals using the dispatcher.

The easiest way to run some code on a background thread is to use the task parallel library (TPL) to start a new task:

Task.Run(()=> 
        {
            for (int i = 1; i <= 9999; i++)
            {
                System.Threading.Thread.Sleep(1500); //simulate long-running operation by sleeping for 1.5s seconds... 
                label1.Dispatcher.BeginInvoke(new Action(() => label1.Content = 100 * i / 9999 + "%"));
            }
        });

my message box immediately show message after i run command while percentage is running

As mentioned the Task is being executed on another thread. So it will run in parallel with the UI thread. That's the reason why the UI can be updated while the loop is running.

You could use the Task.ContinueWith method to show the MessageBox after the task has completed:

int i = 1;
Task.Run(() =>
            {
                for (; i <= 9999; i++)
                {
                    System.Threading.Thread.Sleep(1500); //simulate long-running operation by sleeping for 1.5s seconds... 
                    label1.Dispatcher.BeginInvoke(new Action(() => label1.Content = (i / 9999) * 100 + "%"));
                }
            }).ContinueWith(t =>
            {
              MessageBox.Show("done..." + i.ToString());
            }, System.Threading.CancellationToken.None, TaskContinuationOptions.None, TaskScheduler.FromCurrentSynchronizationContext());
3
On

thats because you are blocking UI thread while calculating inside your loop. Such things should be made in different thread with callback method.

As a work around, you can import system.windows.forms.dll assembly and call System.Windows.Forms.Application.DoEvents();

    for (int i = 1; i <= 9999; i++)
    {
        System.Windows.Forms.Application.DoEvents();
        // do something
        label1.Content = 100 * i / 9999 + "%";
        Thread.Sleep(10);
    }

But this is a bad practice!

0
On

@mm8 solution is ok, but every concurrency round trip to the UI thread from the new one will delay your loop.

If you want the loop ends as early as possible without bloating the UI (and your targeting NET Framework 4.5 or above), you can use the asynchronous Task method equivalent to the old DoEvents: Task.Yield().

    private async void Button_Click(object sender, RoutedEventArgs e)
    {
        for (int i = 0; i < 10000; i++)
        {
            //Do something
            //Simulate work with Task.Delay(millisecondos)
            Label1.Content = (i * 100 / 10000) + "%";
            await Task.Yield(); //give opportunity to dispatch other events
        }
    }

This solution is simple and is an example of why asynchrony doesn't need multithrading always as stated in this answer and this

But there is a trade off between give a break with Task.Yield() and create a new Thread as in @mm8 solution:

  1. Give a break with Task.Yield() is better in case the work can be done in pieces and in each piece the UI must be updated.
  2. Create a new thread is better when UI updates are few (relatively to work done)

NOTE. By the way, in a 10000 loop, I believe is better to show a per thousand progress