質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

新規登録して質問してみよう
ただいま回答率
85.48%
C#

C#はマルチパラダイムプログラミング言語の1つで、命令形・宣言型・関数型・ジェネリック型・コンポーネント指向・オブジェクティブ指向のプログラミング開発すべてに対応しています。

非同期処理

非同期処理とは一部のコードを別々のスレッドで実行させる手法です。アプリケーションのパフォーマンスを向上させる目的でこの手法を用います。

Q&A

解決済

2回答

1744閲覧

[C#]非同期メソッド内のスレッド番号について

tsumayoji13

総合スコア3

C#

C#はマルチパラダイムプログラミング言語の1つで、命令形・宣言型・関数型・ジェネリック型・コンポーネント指向・オブジェクティブ指向のプログラミング開発すべてに対応しています。

非同期処理

非同期処理とは一部のコードを別々のスレッドで実行させる手法です。アプリケーションのパフォーマンスを向上させる目的でこの手法を用います。

0グッド

1クリップ

投稿2022/01/19 01:37

非同期メソッド内のスレッド番号について

前提・実現したいこと

非同期メソッドの中で、タスク外の処理はメインスレッド(番号1)で行われ、
タスク内はサブスレッド(番号3)で行われるものだと思ったのですが、
スレッド番号を取得してみたらタスク後の処理(SampleAsync完了と表示させる部分)がサブスレッドで行われているように見えました。
メソッド内でタスクが作られた場合それ以降はすべて非同期扱いになるのでしょうか?
あと必ず別スレッドの番号が3になるのですが2はないのでしょうか?

該当のソースコード

C#

1 static void Main(string[] args) 2 { 3 int mainId = Thread.CurrentThread.ManagedThreadId; 4 5 Console.WriteLine($"ID:{mainId}で開始"); 6 Console.WriteLine(1); 7 Console.WriteLine(2); 8 Task tsk = SampleAsync(); 9 Console.WriteLine(3); 10 Console.WriteLine(4); 11 tsk.Wait(); 12 Console.WriteLine(5); 13 Console.WriteLine(6); 14 Console.WriteLine($"ID:{mainId}で終了"); 15 16 Console.ReadLine(); 17 } 18 19 static async Task SampleAsync() 20 { 21 int mainThreadNumber = Thread.CurrentThread.ManagedThreadId; 22 Console.WriteLine($"ID:{mainThreadNumber}でSampleAsync開始..."); 23 24 await Task.Run(() => 25 { 26 int taskThreadNumber = Thread.CurrentThread.ManagedThreadId; 27 for (int i = 0; i <= 5; i++) 28 { 29 Console.WriteLine($"ID:{taskThreadNumber}で非同期実行中...[{i}]"); 30 } 31 }).ConfigureAwait(false); 32 33 int taskAfterNumber = Thread.CurrentThread.ManagedThreadId;//←のスレッド番号が3になっている 34 Console.WriteLine($"ID:{taskAfterNumber}でSampleAsync完了..."); 35 } 36

出力結果

C#

1ID:1で開始 21 32 4ID:1でSampleAsync開始...←スレッド番号1で開始している 53 64 7ID:3で非同期実行中...[0] 8ID:3で非同期実行中...[1] 9ID:3で非同期実行中...[2] 10ID:3で非同期実行中...[3] 11ID:3で非同期実行中...[4] 12ID:3で非同期実行中...[5] 13ID:3でSampleAsync完了...←タスク外だがスレッド番号が3になっている 145 156 16ID:1で終了

気になる質問をクリップする

クリップした質問は、後からいつでもMYページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

退会済みユーザー

退会済みユーザー

2022/01/19 01:57

質問に書かれているのはコンソールアプリですよね? コンソールアプリと WinForms, WPF, ASP.NET とは非同期の場合のスレッドの使い方が違ってきますが、コンソールアプリ前提の話で良いのですか?
guest

回答2

0

ベストアンサー

質問に書かれているのはコンソールアプリですよね? コンソールアプリと WinForms, WPF, ASP.NET とは非同期の場合のスレッドの使い方が違ってきますが、コンソールアプリ前提の話で良いと理解して・・・

コンソールアプリ / WinForms, WPF などの GUI アプリ / ASP.NET Web アプリでは非同期操作の際に使われる SynchronizationContext が違うそうで、それによりスレッドの使われ方が違ってきます。

詳しくは以下の記事の 1 ~ 8 項の説明に書きましたので見てください。(自分の独断に基づく解釈も混じっていますか、大筋では間違いないと思っています)

SynchronizationContext とは?
http://surferonwww.info/BlogEngine/post/2020/09/30/what-is-synchronizationcontext.aspx

なので、質問者さんのコンソールアプリの場合は納得のスレッドの使われ方だと思います。

投稿2022/01/19 02:11

編集2022/01/19 02:12
退会済みユーザー

退会済みユーザー

総合スコア0

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

tsumayoji13

2022/01/19 09:33

コンソールやフォームアプリケーションで仕様が変わるんですね・・・コンソールでデッドロックは起きないのもこれに起因する感じでしょうか。 非同期処理は分からないことだらけなので色々見てみたいと思います。
guest

0

あと必ず別スレッドの番号が3になるのですが2はないのでしょうか?

ManagedThreadId = 2 は GC ですね。

C#

1using System; 2using System.Threading; 3 4class Program 5{ 6 static void Main(string[] args) { 7 PrintId(); 8 var test = new GCTest(); 9 test = null; 10 GC.Collect(); 11 Console.ReadKey(); 12 } 13 14 static void PrintId() { 15 Console.WriteLine($"{Thread.CurrentThread.ManagedThreadId}"); 16 } 17 18 private class GCTest 19 { 20 ~GCTest() { 21 PrintId(); 22 } 23 } 24} 25 26結果 271 282 29

投稿2022/01/19 04:55

KOZ6.0

総合スコア2626

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

tsumayoji13

2022/01/19 09:34

ガベージコレクションの存在を忘れていました。ここでGCが働いているのですね。 疑問がスッキリしました。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
85.48%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問