Consider a Winforms application, where we have a button that generates some results. If the user presses the button a second time, it should cancel the first request to generate results and start a new one.
We're using the below pattern, but we are unsure if some of the code is necessary to prevent a race condition (see the commented out lines).
private CancellationTokenSource m_cts;
private void generateResultsButton_Click(object sender, EventArgs e)
{
// Cancel the current generation of results if necessary
if (m_cts != null)
m_cts.Cancel();
m_cts = new CancellationTokenSource();
CancellationToken ct = m_cts.Token;
// **Edit** Clearing out the label
m_label.Text = String.Empty;
// **Edit**
Task<int> task = Task.Run(() =>
{
// Code here to generate results.
return 0;
}, ct);
task.ContinueWith(t =>
{
// Is this code necessary to prevent a race condition?
// if (ct.IsCancellationRequested)
// return;
int result = t.Result;
m_label.Text = result.ToString();
}, ct, TaskContinuationOptions.OnlyOnRanToCompletion, TaskScheduler.FromCurrentSynchronizationContext());
}
Notice:
- We only ever cancel the
CancellationTokenSource
on the main thread. - We use the same
CancellationToken
in the continuation as we do in the original task.
We're wondering whether or not the following sequence of events is possible:
- User clicks "generate results" button. Initial task t1 is started.
- User clicks "generate results" button again. Windows message is posted to queue, but the handler hasn't been executed yet.
- Task t1 finishes.
- TPL
startsprepares to start the continuation (since theCancellationToken
is not cancelled yet). The task scheduler posts the work to the Windows message queue (to get it to run on the main thread). - The generateResultsButton_Click for the 2nd click starts executing and the
CancellationTokenSource
is cancelled. - The continuations work starts and it operates as though the token were not cancelled (i.e. it displays its results in the UI).
So, I think the question boils down to:
When work is posted to the main thread (by using TaskScheduler.FromCurrentSynchronizationContext()
) does the TPL check the CancellationToken
on the main thread before executing the task's action, or does it check the cancellation token on whatever thread it happens to be on, and then post the work to the SynchronizationContext
?