注:エラーは、WaitHandle.WaitAll(doneEvents);行にあります。標準のWPFプロジェクトを使用しています。
private void Search()
{
const int CPUs = 2;
var doneEvents = new ManualResetEvent[CPUs];
// Configure and launch threads using ThreadPool:
for (int i = 0; i < CPUs; i++)
{
doneEvents[i] = new ManualResetEvent(false);
var f = new Indexer(Paths[i], doneEvents[i]);
ThreadPool.QueueUserWorkItem(f.WaitCallBack, i);
}
// Wait for all threads in pool
WaitHandle.WaitAll(doneEvents);
Debug.WriteLine("Search completed!");
}
更新:次のソリューションはWPFアプリケーションでは機能しません!メインアプリケーション属性をMTAThreadAttributeに変更することはできません。次のエラーが発生します。
エラー: "STAスレッド上の複数のハンドルに対するWaitAllはサポートされていません。"
タスクを使用してスレッド化を行うのはどうですか。
http://msdn.Microsoft.com/en-us/library/system.threading.tasks.task.aspx
var task1 = Task.Factory.StartNew(() => DoSomeWork());
var task2 = Task.Factory.StartNew(() => DoSomeWork());
var task3 = Task.Factory.StartNew(() => DoSomeWork());
Task.WaitAll(task1, task2, task3);
実際、以下を使用してWaitHandle.WaitAll(doneEvents);を置き換えます。
foreach (var e in doneEvents)
e.WaitOne();
ManualResetEvent
を1つ使用して待機します。また、起動するワーカースレッドの数に設定されたTaskCount変数を維持し、ワーカーの最後のアクションとしてワーカースレッドコードでInterlocked.Decrement
を使用し、カウンターがゼロに達した場合にイベントを通知します。
// other worker actions...
if (Interlocked.Decrement(ref taskCount) == 0)
doneEvent.Set();
代わりにCountdownEvent
クラスを使用するようにコードをリファクタリングします。
private void Search()
{
const int CPUs = 2;
var done = new CountdownEvent(1);
// Configure and launch threads using ThreadPool:
for (int i = 0; i < CPUs; i++)
{
done.AddCount();
var f = new Indexer(Paths[i], doneEvents[i]);
ThreadPool.QueueUserWorkItem(
(state) =>
{
try
{
f.WaitCallBack(state);
}
finally
{
done.Signal();
}
}, i);
}
// Wait for all threads in pool
done.Signal();
done.Wait();
Debug.WriteLine("Search completed!");
}
次のようなものを使用します。
foreach (ITask Task in Tasks)
{
Task.WaitHandle = CompletedEvent;
new Thread(Task.Run).Start();
}
int TasksCount = Tasks.Count;
for (int i = 0; i < TasksCount; i++)
CompletedEvent.WaitOne();
if (AllCompleted != null)
AllCompleted(this, EventArgs.Empty);