web-dev-qa-db-ja.com

STAスレッド上の複数のハンドルに対するWaitAllはサポートされていません

  1. なぜこのエラーメッセージが表示されるのですか? 「STAスレッド上の複数のハンドルに対するWaitAllはサポートされていません。」
  2. [MTAThreadAttribute]属性を使用する必要がありますか? 更新: WPFアプリケーションでは動作しません!

注:エラーは、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はサポートされていません。"

45
Amir Rezaei

タスクを使用してスレッド化を行うのはどうですか。

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);
20
Oliver

実際、以下を使用してWaitHandle.WaitAll(doneEvents);を置き換えます。

foreach (var e in doneEvents)
    e.WaitOne();
52
Calimero100582

ManualResetEventを1つ使用して待機します。また、起動するワーカースレッドの数に設定されたTaskCount変数を維持し、ワーカーの最後のアクションとしてワーカースレッドコードでInterlocked.Decrementを使用し、カウンターがゼロに達した場合にイベントを通知します。

// other worker actions...
if (Interlocked.Decrement(ref taskCount) == 0)
   doneEvent.Set();
10
liggett78

代わりに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!"); 
} 
7
Brian Gideon

次のようなものを使用します。

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);
0
CSharper