Upgrade to Pro
— share decks privately, control downloads, hide ads and more …
Speaker Deck
Features
Speaker Deck
PRO
Sign in
Sign up for free
Search
Search
【Crystal】Concurrency
Search
Sponsored
·
Your Podcast. Everywhere. Effortlessly.
Share. Educate. Inspire. Entertain. You do you. We'll handle the rest.
→
at_grandpa
July 18, 2018
Technology
3
910
【Crystal】Concurrency
crystal.tokyo #7 での発表資料です。
https://crystal.connpass.com/event/93629/
at_grandpa
July 18, 2018
Tweet
Share
More Decks by at_grandpa
See All by at_grandpa
技術書典5でチャレンジしたあれこれ
atgrandpa
3
2k
CSS組版おもしろい!
atgrandpa
1
330
Rubyを好きになった過程
atgrandpa
3
1.6k
技術同人誌を執筆して得たこと
atgrandpa
3
1.6k
【Crystal】Macroについて
atgrandpa
3
1.5k
初のサークル参加で気づいた小さなこと
atgrandpa
0
500
Other Decks in Technology
See All in Technology
IBM Bobを使って、PostgreSQLのToDoアプリをDb2へ変換してみよう/202603_Dojo_Bob
mayumihirano
1
330
生成AIの利用とセキュリティ /gen-ai-and-security
mizutani
1
1.7k
AI実装による「レビューボトルネック」を解消する仕様駆動開発(SDD)/ ai-sdd-review-bottleneck
rakus_dev
0
120
身体を持ったパーソナルAIエージェントの 可能性を探る開発
yokomachi
1
110
Shifting from MCP to Skills / ベストプラクティスの変遷を辿る
yamanoku
4
830
親子 or ペアで Mashup for the Future! しゃべって楽しむ 初手AI駆動でものづくり体験
hiroramos4
PRO
0
110
わからなくて良いなら、わからなきゃだめなの?
kotaoue
1
330
作りっぱなしで終わらせない! 価値を出し続ける AI エージェントのための「信頼性」設計 / Designing Reliability for AI Agents that Deliver Continuous Value
aoto
PRO
2
290
OSC仙台プレ勉強会 AlmaLinuxとは
koedoyoshida
0
150
JAWS FESTA 2025でリリースしたほぼリアルタイム文字起こし/翻訳機能の構成について
naoki8408
1
440
DevOpsエージェントで実現する!! AWS Well-Architected(W-A) を実現するシステム設計 / 20260307 Masaki Okuda
shift_evolve
PRO
3
690
AI時代の「本当の」ハイブリッドクラウド — エージェントが実現した、あの頃の夢
ebibibi
0
110
Featured
See All Featured
Impact Scores and Hybrid Strategies: The future of link building
tamaranovitovic
0
230
Applied NLP in the Age of Generative AI
inesmontani
PRO
4
2.2k
Getting science done with accelerated Python computing platforms
jacobtomlinson
2
140
Utilizing Notion as your number one productivity tool
mfonobong
4
260
B2B Lead Gen: Tactics, Traps & Triumph
marketingsoph
0
76
A brief & incomplete history of UX Design for the World Wide Web: 1989–2019
jct
1
320
Exploring anti-patterns in Rails
aemeredith
2
290
How To Stay Up To Date on Web Technology
chriscoyier
790
250k
Introduction to Domain-Driven Design and Collaborative software design
baasie
1
640
The State of eCommerce SEO: How to Win in Today's Products SERPs - #SEOweek
aleyda
2
9.9k
Testing 201, or: Great Expectations
jmmastey
46
8.1k
State of Search Keynote: SEO is Dead Long Live SEO
ryanjones
0
150
Transcript
Concurrency 2018.07.18 @at_grandpa Crystal.tokyo #7 in 渋谷
@at_grandpa
圧倒亭グランパのブログ
Concurrency
✔ Concurrency の雰囲気を話します ✔ 細かい syntax などはドキュメント参照
None
“as in Go or Clojure” ✔ Go の goroutine/channel とほぼ同じ
終了!
Concurrency の中身を ちょっと覗いてみましょう
サンプルコード ✔ 並行処理 ✔ 各処理からの値の取得
None
String型のChannelをインスタンス化
2つのFiberを生成
Channel経由で値を取得
2018-07-18 08:12:50 +09:00 start 2018-07-18 08:12:50 +09:00 [fiber 1] start
2018-07-18 08:12:50 +09:00 [fiber 2] start 2018-07-18 08:12:55 +09:00 [top level] value: send from fiber 1 2018-07-18 08:12:55 +09:00 [fiber 1] end 2018-07-18 08:13:00 +09:00 [top level] value: send from fiber 2 2018-07-18 08:13:00 +09:00 end
2018-07-18 08:12:50 +09:00 start 2018-07-18 08:12:50 +09:00 [fiber 1] start
2018-07-18 08:12:50 +09:00 [fiber 2] start 2018-07-18 08:12:55 +09:00 [top level] value: send from fiber 1 2018-07-18 08:12:55 +09:00 [fiber 1] end 2018-07-18 08:13:00 +09:00 [top level] value: send from fiber 2 2018-07-18 08:13:00 +09:00 end ・並行に動いている ・値も取れている
どういう仕組みで動いているか
✔ Fiber ✔ Runtime Scheduler ✔ Event Loop ✔ Channel
✔ IO::Syscall Concurrencyを理解するポイント
Fiber ✔ Process ⊃ Thread ⊃ Fiber ✔ 協調マルチタスク -
Fiber自ら、処理を他のFiberに委譲する - 1つのFiberが固まるとシステム全体が固まる ✔ Crystalの処理は全てFiberで行われている - 「Main Fiber」でメインの処理を実行している
Runtime Scheduler ✔ Fiberの切り替えを担当 ✔ クラス変数にFiberのqueueを持っている - @@runnables = Deque(Fiber).new
- 実行可能Fiberのqueue ✔ Scheduler.rescheduleで切り替え
Event Loop ✔ I/O処理の委譲先 ✔ 委譲している間に別のFiberを実行できる ✔ I/O処理が終了したら委譲元のFiberに移る
Channel ✔ Fiber間のデータのやりとり ✔ 送信元Fiberや受信先Fiberを保持 - @senders = Deque(Fiber).new -
@receivers = Deque(Fiber).new ✔ 送受信時にFiberを切り替え - Runtime Scheduler を使う
IO::Syscall ✔ 以下でincludeされている - Crystal::System::FileDescriptor - Socket ✔ read/writeでFiber切り替え -
Runtime Scheduler を使う ほぼ全てのI/Oを網羅
実際の動き追う
None
Channelをインスタンス化
・Fiber1を定義 ・Runtime Scheduler の 実行可能Fiberのqueueに Fiber1が enqueue される
concurrency.cr ・Fiber1を定義 ・Runtime Scheduler の 実行可能Fiberのqueueに Fiber1が enqueue される
・Fiber2を定義 ・Runtime Scheduler の 実行可能Fiberのqueueに Fiber2が enqueue される
・実際の処理はここから開始 ・Top Level のコードは 「Main Fiber」で動いている ・Channelの送受信時には Fiberの切り替えが行われる
・実際の処理はここから開始 ・Top Level のコードは 「Main Fiber」で動いている ・Channelの送受信時には Fiberの切り替えが行われる ・Runtime Scheduler
の 実行可能Fiberのqueue からshift ・Fiber1に処理が移る
・Fiber1に処理が移った
・sleep
concurrency.cr
concurrency.cr fiber.cr
concurrency.cr fiber.cr ・Event Loop に処理を委譲 ・Runtime Scheduler で queueの次のFiber切り替え
concurrency.cr fiber.cr ・Event Loop に処理を委譲 ・Runtime Scheduler で queueの次のFiber切り替え ・Fiber2へ移る
・Fiber2に処理が移った
・同じくFiber切り替え ・しかし、もう 実行可能Fiberのqueueには Fiberが存在しない ・I/O処理を待機するしかない
待機中 . . .
・Event Loop が sleep 5 の 終了を検知 ・処理中のFiberが他にいないので Fiber1の処理が再開される
・Channelに値を送信 ・sendの場合はreceiveを 呼んだFiberに切り替え ・Main Fiber に切り替わる ・このとき、sendしたFiberを Schedulerのqueueにenqueue
・Channelに値を送信 ・sendの場合はreceiveを 呼んだFiberに切り替え ・Main Fiber に切り替わる ・このとき、sendしたFiberを Schedulerのqueueにenqueue
・受信された値を表示
・Schedulerの 実行可能Fiberのqueueを元に Fiberの切り替え ・queueにはsendした際に enqueueされたFiber1が入っている
・Schedulerの 実行可能Fiberのqueueを元に Fiberの切り替え ・queueにはsendした際に enqueueされたFiber1が入っている ・Fiber1が再開
・値の表示
・Fiber1のブロックが終了 ・Fiberの切り替えが発生 ・しかし、Schedulerの queueには実行可能Fiberが 存在しない
待機中 . . .
・Event Loop が sleep 10 の 終了を検知 ・処理中のFiberが他にいないので Fiber2に処理が戻る
・Channelに値を送信 ・sendの場合はreceiveを 呼んだFiberに切り替え ・Main Fiber に切り替わる ・このとき、sendしたFiberを Schedulerのqueueにenqueue
・Channelに値を送信 ・sendの場合はreceiveを 呼んだFiberに切り替え ・Main Fiber に切り替わる ・このとき、sendしたFiberを Schedulerのqueueにenqueue
・受信された値を表示
・そのまま終了
・そのまま終了 ・ここは通らない
・そのまま終了 ・ここは通らない 2018-07-18 08:12:50 +09:00 start 2018-07-18 08:12:50 +09:00 [fiber
1] start 2018-07-18 08:12:50 +09:00 [fiber 2] start 2018-07-18 08:12:55 +09:00 [top level] value: send from fiber 1 2018-07-18 08:12:55 +09:00 [fiber 1] end 2018-07-18 08:13:00 +09:00 [top level] value: send from fiber 2 2018-07-18 08:13:00 +09:00 end [fiber 2] end は表示されてない
複雑!
とはいえ ✔ Concurrencyに必要な役者を知る - Fiber, Runtime Scheduler, Event loop, Channel
✔ Fiberが切り替わるタイミングを知る - I/Oの場合 - 実行可能Fiber-queueからshift - receiveの場合 - 実行可能Fiber-queueからshift - sendしたFiberをqueueにenqueue - sendの場合 - receiveしたFiberに切り替え これらを知るだけで、だいぶ変わる
まとめ
✔ 並行処理 ✔ Concurrencyに必要な役者を知ろう ✔ Fiberを切り替えるタイミングを知ろう Concurrency
✔ 立て続けにsendされたらどうなるの? - sendされた値もqueueに保存される - receiveを呼ぶ度にqueueからshift ✔ Fiber内でI/O以外の重い処理があったら? - 委譲できない処理はそのまま処理される
- その処理が終わるまで他のFiberは実行できない ✔ 入れ子spawnとかどうなるんだろう? - \(^o^)/ まだまだあるよ
Happy Crystalling ! fin