Я новичок в TPL и WPf и столкнулся со следующей проблемой. Я пытаюсь загрузить сайт в бесконечном цикле (здесь только цикл for) и добавить его в очередь. Следующая задача берет его и показывает в текстовом блоке. Однако я, кажется, не понимаю правильный поток для пользовательского интерфейса, хотя я думаю, что правильно использую TaskScheduler.
Спасибо за любую помощь!
BlockingCollection<string> blockingCollection = new BlockingCollection<string>();
CancellationToken token = tokenSource.Token;
TaskScheduler uiScheduler = TaskScheduler.FromCurrentSynchronizationContext();
Task task1 = new Task(
(obj) =>
{
for (int i = 0; i < 10; i++)
{
if (token.IsCancellationRequested)
{
TxtBlock2.Text = "Task cancel detected";
throw new OperationCanceledException(token);
}
else
{
string code = i.ToString() + "\t" + AsyncHttpReq.get_source_WebRequest(uri);
blockingCollection.Add(code);
}
}
}, TaskScheduler.Default);
task1.ContinueWith(antecedents =>
{
TxtBlock2.Text = "Signalling production end";
blockingCollection.CompleteAdding();
}, uiScheduler);
Task taskCP = new Task(
(obj) =>
{
while (!blockingCollection.IsCompleted)
{
string dlCode;
if (blockingCollection.TryTake(out dlCode))
{
//the calling thread cannot access this object because a different thread owns it.
TxtBlock3.Text = dlCode;
}
}
}, uiScheduler);
WindowsBase.dll!System.Windows.Threading.Dispatcher.VerifyAccess() + 0x4a bytes
WindowsBase.dll!System.Windows.DependencyObject.SetValue(System.Windows.DependencyProperty dp, object value) + 0x19 bytes
PresentationFramework.dll!System.Windows.Controls.TextBlock.Text.set(string value) + 0x24 bytes
WpfRibbonApplication4.exe!WpfRibbonApplication4.MainWindow.Button1_Click.AnonymousMethod__4(object obj) Строка 83 + 0x16 байт C# mscorlib.dll!System.Threading.Tasks.Task.InnerInvoke() + 0x44 байт mscorlib.dll!System.Threading.Tasks.Task .Execute() + 0x43 байта mscorlib.dll!System.Threading.Tasks.Task.ExecutionContextCallback(object obj) + 0x27 байт
mscorlib.dll!System.Threading.ExecutionContext.Run(System.Threading.ExecutionContext executeContext, System Обратный вызов .Threading.ContextCallback, состояние объекта, bool ignoreSyncCtx) + 0xb0 байт
mscorlib.dll!System.Threading.Tasks.Task.ExecuteWithThreadLocal(ref System.Threading.Tasks.Task currentTaskSlot) + 0x154 байт
mscorlib. dll!System.Threading.Tasks.Task.ExecuteEntry(bool bPreventDoubleExecution) + 0x8b байт
mscorlib.dll!System.Threading.Tasks.Task.System.Threading.IThreadPoolWorkItem.ExecuteWorkItem() + 0x7 байт mscorlib.dll!System .Threading.ThreadPoolWorkQueue.Disp atch() + 0x147 байт
mscorlib.dll!System.Threading._ThreadPoolWaitCallback.PerformWaitCallback() + 0x2d байт
[переход из исходного состояния в управляемый]
System.InvalidOperationException was unhandled by user code
Message=The calling thread cannot access this object because a different thread owns it.
Source=WindowsBase
StackTrace:
at System.Windows.Threading.Dispatcher.VerifyAccess()
at System.Windows.DependencyObject.SetValue(DependencyProperty dp, Object value)
at System.Windows.Controls.TextBlock.set_Text(String value)
at WpfRibbonApplication4.MainWindow.<>c__DisplayClass5.<Button1_Click>b__3(Object o) in C:\ ... \WpfRibbonApplication4\WpfRibbonApplication4\MainWindow.xaml.cs:line 90
at System.Threading.Tasks.Task.InnerInvoke()
at System.Threading.Tasks.Task.Execute()
InnerException:
Большое спасибо за вашу помощь. У меня все еще есть два вопроса: я немного переписал свой код с помощью Task.Factory.StartNew. Однако моя Task2, похоже, вызывает проблемы. Там нет сообщения об ошибке. Похоже на тугую петлю. Я, конечно, не понял, почему? Не могли бы вы быть так любезны и снова указать мне правильное направление. Имейте в виду, что я занимаюсь C# примерно 6 месяцев, а TPL неделю, иначе я бы не стал вас снова спрашивать. Но с таким количеством опыта... Еще раз спасибо!
Код Брайанса:
var task1 = new Task(
(obj) =>
Зачем нужен obj
?
private void Button1_Click(object sender, RoutedEventArgs e)
{
TaskScheduler uiTaskScheduler = TaskScheduler.FromCurrentSynchronizationContext(); BlockingCollection blockingCollection = new BlockingCollection(); CancellationTokenSource cts = новый CancellationTokenSource();
CancellationToken token = cts.Token;
Task task1 = Task.Factory.StartNew(
() =>
{
for (int i = 0; i < 10 ; i++)
{
token.ThrowIfCancellationRequested();
string code = i++.ToString() + "\t" + AsyncHttpReq.get_source_WebRequest(uriDE);
blockingCollection.Add(code);
}
}, token, TaskCreationOptions.None, TaskScheduler.Default);
task1.ContinueWith(
(antecedents) =>
{
if (token.IsCancellationRequested)
{
TxtBlock2.Text = "Task cancel detected";
}
else
{
TxtBlock2.Text = "Signalling production end";
}
blockingCollection.CompleteAdding();
}, uiTaskScheduler);
Task task2 = Task.Factory.StartNew(
() =>
{
while (!blockingCollection.IsCompleted)
{
string dlcode;
if (blockingCollection.TryTake(out dlcode))
{
TxtBlock3.Text = dlcode;
}
}
}, token, TaskCreationOptions.None, uiTaskScheduler);
}