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

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

新規登録して質問してみよう
ただいま回答率
85.35%
Go

Go(golang)は、Googleで開発されたオープンソースのプログラミング言語です。

Q&A

1回答

1792閲覧

チャネル, goルーチンを使った時の処理の流れをご教示いただきたいです

yaha4967

総合スコア106

Go

Go(golang)は、Googleで開発されたオープンソースのプログラミング言語です。

0グッド

0クリップ

投稿2021/09/24 05:01

以下のコードなのですが

fibonacciのforループ一週目でcチャネルに値が入る

goルーチンのforループが一つ動く

fibonacciのforループ2週目でcチャネルに値が入る

goルーチンのforループがまた一つ動く


という流れなのかな?と思ったのですが出力を見た感じ違うようです。どのような流れの処理になっているのでしょうか?

go

1package main 2 3import "fmt" 4 5func fibonacci(c, quit chan int) { 6 x, y := 0, 1 7 8 for { 9 select { 10 case c <- x: 11 x, y = y, x+y 12 fmt.Println("フィボナッチの方") 13 case <-quit: 14 fmt.Println("quit") 15 return 16 } 17 } 18} 19 20func main() { 21 c := make(chan int) 22 quit := make(chan int) 23 go func() { 24 for i := 0; i < 10; i++ { 25 fmt.Println(<-c) 26 fmt.Println("Goルーチンの方") 27 28 } 29 quit <- 8 30 }() 31 fibonacci(c, quit) 32}

実行結果

0 Goルーチンの方 フィボナッチの方 フィボナッチの方 1 Goルーチンの方 1 Goルーチンの方 フィボナッチの方 フィボナッチの方 2 Goルーチンの方 3 Goルーチンの方 フィボナッチの方 フィボナッチの方 5 Goルーチンの方 8 Goルーチンの方 フィボナッチの方 フィボナッチの方 13 Goルーチンの方 21 Goルーチンの方 フィボナッチの方 フィボナッチの方 34

よろしくお願いします。

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

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

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

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

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

guest

回答1

0

これチャネルの挙動をよく知っていないと理解しにくいです。
mainのgoroutine(送信側)とmain関数内で起こしたgoroutine(受信側)の2つが切り替わりながら動作します。

  • チャネルの受信待ちに入る時、未送信待ちの場合は所属goroutineをブロックしgoroutineスイッチする
  • チャネルの受信完了でgoroutine再開対象になる
  • チャネルの受信待ちに入る時、送信待ち済みの場合は直ちに受信してgoroutineスイッチしません
  • チャネルの送信待ちに入る時、未受信待ちの場合は所属goroutineをブロックしgoroutineスイッチする
  • チャネルの送信完了でgoroutine再開対象になる
  • チャネルの送信待ちに入る時、受信待ち済みの場合は直ちに送信してgoroutineスイッチしません
  • goroutine再開対象が複数ある場合はランダムなgoroutineが選ばれて動作を再開する(今回の例では2つしかgoroutineがないので必ずもう一つのgoroutineが動作を再開する)

case c <- x:S<-cRと記述してこのコードでどういう順番が発生しているのかを列挙してみましょう。

  1. 初回のS送信待ちに入る(goroutineスイッチ)
  2. 初回のRは送信待ち済みのため直接R受信して「Goルーチンの方」が表示される
  3. 次のR受信待ちに入る(goroutineスイッチ)
  4. 送信側の処理が再開「フィボナッチの方」が表示される
  5. 次のSは受信待ち済みなので直接S送信して「フィボナッチの方」が表示される
  6. 次のS送信待ちに入る(goroutineスイッチ)
  7. 受信側の処理が再開「Goルーチンの方」が表示される
  8. 次のRは送信待ち済みのため直接R受信して「Goルーチンの方」が表示される
  9. 3.に戻って繰り返す

結果としてそれぞれ2回ずつの表示を繰り返す挙動になります。

投稿2021/09/27 04:20

編集2021/09/27 04:41
nobonobo

総合スコア3367

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

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

yaha4967

2021/09/29 07:30

回答ありがとうございます! まだ理解できてないのですが、ヒントいただけたのでもう少し調べてみます!
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

まだベストアンサーが選ばれていません

会員登録して回答してみよう

アカウントをお持ちの方は

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

ただいまの回答率
85.35%

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

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

質問する

関連した質問