web-dev-qa-db-ja.com

スレッドのリストを取得する

List<>クラスを使用せずに、実行中のすべてのスレッドを一覧表示したい。実行中のスレッドを動的に監視したい。どうやってやるの?

39
sanchop22
using System.Diagnostics;

ProcessThreadCollection currentThreads = Process.GetCurrentProcess().Threads;

foreach (ProcessThread thread in currentThreads)    
{
   // Do whatever you need
}
52
Xaqron

方法1:OSスレッドを取得する

これにより、OSスレッドのリストが取得されます。

ProcessThreadCollection currentThreads = Process.GetCurrentProcess().Threads;

foreach (ProcessThread thread in currentThreads)
{
}

方法2:マネージスレッドを取得する

マネージスレッドはOSスレッドの上にあります。 IDは異なり、理論上は、複数のマネージスレッドが1つのOSスレッドの上にある場合があります(実際にはこれを観察していませんが)。

マネージスレッドを取得することは、実際に行う必要があるよりも難しいことがわかります。

方法2.1:マネージスレッドを取得する最も簡単なコード

  1. GitHubのMicrosoft.Diagnostics.Runtime を確認してください。
  2. NuGetパッケージをインストールします CLR Memory Diagnostics(ClrMD)

次に、前述のNuGetパッケージを使用して独自のプロセスにアタッチし、マネージスレッドを読み取ります。

using Microsoft.Diagnostics.Runtime;

using (DataTarget target = DataTarget.AttachToProcess(
    Process.GetCurrentProcess().Id, 5000, AttachFlag.Passive))
{
    ClrRuntime runtime = target.ClrVersions.First().CreateRuntime();
    foreach (ClrThread thread in runtime.Threads)
    {
    }
}

方法2.2:スタックトレースによってマネージスレッドを検索する方法の例

残念ながら、スレッド名でスレッドのリストを検索する方法を見つけることができませんでした。

ただし、すべてが失われるわけではありません。マネージスレッドを作成し、スタックフレームを検索して名前空間の一致を見つけ、そのプロパティを出力する方法の例を次に示します。

namespace MyTest
{
    int managedThreadId = 0;
    var task = Task.Run(
        () =>
        {
            // Unfortunately, cant see "Testing" anywhere in result returned
            // from NuGet package ClrMD ...
            Thread.CurrentThread.Name = "Testing";
            Thread.Sleep(TimeSpan.FromDays(1));
        });


    // ... so we look for our thread by the first Word in this namespace.
    string startOfThisNamespace = this.GetType().Namespace.ToString().Split('.')[0]; // Is "MyTest".
    using (DataTarget target = DataTarget.AttachToProcess(Process.GetCurrentProcess().Id, 5000, AttachFlag.Passive))
    {
        ClrRuntime runtime = target.ClrVersions.First().CreateRuntime();

        foreach (ClrThread thread in runtime.Threads)
        {
            IList<ClrStackFrame> stackFrames = thread.StackTrace;

            List<ClrStackFrame> stackframesRelatedToUs = stackFrames
                .Where(o => o.Method != null && o.Method.ToString().StartsWith(startOfThisNamespace)).ToList();

            if (stackframesRelatedToUs.Count > 0)
            {
                Console.Write("ManagedThreadId: {0}, OSThreadId: {1}, Thread: IsAlive: {2}, IsBackground: {3}:\n", thread.ManagedThreadId, thread.OSThreadId, thread.IsAlive, thread.IsBackground);
                Console.Write("- Stack frames related namespace '{0}':\n", startOfThisNamespace);
                foreach (var s in stackframesRelatedToUs)
                {
                    Console.Write("  - StackFrame: {0}\n", s.Method.ToString());
                }
            }
        }
    }
}

作成したスレッド内でManagedThreadIdを保存し、runtime.Threadsでこの同じIDを検索することでも、正しい一致を見つけることができます。

テスト中

以下のすべての組み合わせでテスト済み

  • Visual Studio 2015 SP1
  • .NET 4.5
  • .NET 4.6.0
  • .NET 4.6.1
  • C#5.0
  • C#6.0

参照資料

ランタイムの作成時にClrMdが例外をスローする を参照してください。

49
Contango