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
なぜGoのジェネリクスはこの形なのか? Featherweight Goが明かす設計の核心
Search
Ryotaro
September 28, 2025
Programming
4
250
なぜGoのジェネリクスはこの形なのか? Featherweight Goが明かす設計の核心
https://gocon.jp/2025/talks/958641/
Ryotaro
September 28, 2025
Tweet
Share
Other Decks in Programming
See All in Programming
How Android Uses Data Structures Behind The Scenes
l2hyunwoo
1
540
まだ世にないサービスをAIと創る話 〜 失敗から学ぶフルスタック開発への挑戦 〜
katayamatg
0
140
そのAPI、誰のため? Androidライブラリ設計における利用者目線の実践テクニック
mkeeda
2
5.2k
チームのテスト力を鍛える
goyoki
4
1.2k
エンジニアとして高みを目指す、 利益を生み出す設計の考え方 / design-for-profit
minodriven
16
9k
Platformに“ちょうどいい”責務ってどこ? 関心の熱さにあわせて考える、責務分担のプラクティス
estie
2
450
実践AIチャットボットUI実装入門
syumai
6
2.1k
AccessorySetupKitで実現するシームレスなペアリング体験 / Seamless pairing with AccessorySetupKit
nekowen
0
180
Let's Write a Train Tracking Algorithm
twocentstudios
0
200
Server Less Code More - コードを書かない時代に生きるサーバーレスデザイン / server-less-code-more
gawa
5
1.7k
PostgreSQLで手軽にDuckDBを使う!DuckDB&pg_duckdb入門/osk2025-duckdb
takahashiikki
1
220
Go Conference 2025: Goで体感するMultipath TCP ― Go 1.24 時代の MPTCP Listener を理解する
takehaya
5
390
Featured
See All Featured
StorybookのUI Testing Handbookを読んだ
zakiyama
31
6.1k
個人開発の失敗を避けるイケてる考え方 / tips for indie hackers
panda_program
114
20k
How STYLIGHT went responsive
nonsquared
100
5.8k
How to Think Like a Performance Engineer
csswizardry
27
2k
Automating Front-end Workflow
addyosmani
1371
200k
Put a Button on it: Removing Barriers to Going Fast.
kastner
60
4k
Docker and Python
trallard
46
3.6k
Build your cross-platform service in a week with App Engine
jlugia
231
18k
Product Roadmaps are Hard
iamctodd
PRO
54
11k
CSS Pre-Processors: Stylus, Less & Sass
bermonpainter
358
30k
[RailsConf 2023 Opening Keynote] The Magic of Rails
eileencodes
30
9.7k
How to Create Impact in a Changing Tech Landscape [PerfNow 2023]
tammyeverts
53
3k
Transcript
なぜGoのジェネリクスはこの形なのか? Featherweight Goが明かす設計の核心 CyberAgent / Qualiarts 鈴木 稜太朗 Go Conference
2025
Ryotaro Suzuki •CyberAgent / Qualiarts バックエンドエンジニアとして、運用タイトルの 開発をしている 圏論がとっても好き Go歴は2年くらい Haskellが8年くらい
GOとジェネリクス
ジェネリクスを求める声 Go Developer Survey 2019 Resultsより https://go.dev/blog/survey2019-results
初期案 型が満たすべき条件を定義する contractsの導入 https://go.googlesource.com/proposal/+/master/design/go2draft-contracts.md
想定実装
想定実装
contractsの撤回 • 新しい概念の追加になるので、大変 • すでに「interface」という型の振る舞いを定義できるものがある interfaceをそのまま型制約に上手く使って実装したい
Featherweight Go
Featherweight Go Goチームからの依頼で執筆 ジェネリクス設計において根本の設計思想 ジェリックなコードをどのように変換するか定式化
Featherweight Java
各研究でのモデル FG:Go言語の主要な機能を抽出した最小限の言語モデル FGG:FGにジェネリクスを追加した言語モデル Featherweight Go FJ:Javaの主要な機能を抽出した最小限の言語モデル FGJ:FJにジェネリクスを追加した言語モデル Featherweight Java
Featherweight Goのアプローチ Goチームの要件 • 使いやすく理解しやすいこと • ランタイムコストが低いこと • 可能な限り保守的な拡張であること
方針 • 構造的サブタイピングとジェネリクスの組み合わせ • 追加の言語機能なしに型制約を導入 • モノモーフィゼーションに基づいたコンパイル戦略
FGとFGG
FG • チューニング完全な関数型言語としてモデル化 • Goのコンパクトなサブセット Structs Method Interfaces Structural Typing
• 型アサーション e.(t)
FGモデル
FGの例
FGの課題 EqualにEqを満たした別の型を渡す 実行時にパニック!! ジェネリクスのない世界での多態の限界 『実行時まで安全かわからない』という問題を 解決するのが FGGの設計
FGG • FGのStructs、Methods、Interfacesが型パラメーターを持つ • 型パラメータの制約に Interfacesを使用 • 汎用的な構造に対し、特化した制約を持つ Methodsを定義可能 (共変レ
シーバ) • 構造的サブタイピングを維持
FGG
FGGの例
FGGの比較
FGGの例(再帰的型制約)
FGGの比較
FGGモデル
まとめ FGの世界 • 多態性をインターフェースで表現 • 型の安全性は実行時に型アサーションで保証(プログラマの責任) • 常にパニックのリスクが伴う FGGの世界
• 多態性を型パラメータで表現 • 型の安全性はコンパイル時に型制約で保証(コンパイラの責任) • パニックのリスクをコンパイル時に排除
コンパイル戦略
ジェネリクスのコンパイル戦略 FJ FGJ ジェネリクス イレイジャ FG FGG ジェネリクス モノモーフィゼーション
ジェネリクスのコンパイル戦略 モノモーフィゼーション • 型ごとに使われている部分の専用コードを生成 • List[string] →(コンパイル)→ string専用のList •
ランタイムで型アサーションが制限されない • ランタイムオーバーヘッドが低いが、メモリをより食う イレイジャ • コンパイル時に型情報を削除 • List<String> →(コンパイル)→ List • ランタイムで型アサーションができない • メモリはあまり食わない
変換するFGGのコード
FGGからFGへの変換
FGGからFGへの変換
Pairの場合
多相再帰:モノモルフィゼーションできない 全ての型付け可能なFGGがFGに変換できるわけではない
ダミーメソッド
ダミーメソッド なんのためにダミーメソッドが生成されるのか?
ダミーメソッドの例(List)
ダミーメソッドの例(List) List<bool>のmapは使用しない
ダミーメソッドがない場合 • List[bool]のMapは未使用 • コンパイラはメソッドを省略し、空のイ ンターフェースを生成 • 全ての型が空のインターフェースを満 たす •
コンパイルエラーになるべきlistBool = listIntが可能になり、 型安全性が崩壊
ダミーメソッドの追加
Expression Problem(式問題)
Expression Problem Eval(評価) String(文字列化) PrettyPrint (整形) Num (数) 実装済み 実装済み
後から追加したい Plus (足し算) 実装済み 実装済み 後から追加したい Mul (掛け算) 後から追加したい 後から追加したい 後から追加したい 行(データ型)と列(操作)のテーブル 関数型言語は列の追加が得意、オブジェクト指向言語は行の追加が得意
FGGのアプローチ 「汎用的な構造体定義」と「特化した制約を持つメソッド」の組み合わせ 共変レシーバが、静的な型安全ながら高い拡張性を実現
Goと共変レシーバ リリースされた Goのジェネリクスも不変 (invariant) な設計を採用 - 構造体を定義した時点での型制約が、その後のすべてのメソッドで一貫して適用 - Plus[T any]ならメソッドも
anyのまま、Plus[T Expr]ならメソッドも Exprのまま 現在のGoで、式問題のような拡張性を実現するには • 柔軟性を諦め、静的な安全性を取るか • 静的な安全性を諦め、柔軟性を取るか • 2つのトレードオフに直面
柔軟性の低下
型アサーションの復活
まとめ
まとめ • contractsという新しい概念ではなく、interfaceを土台に変更 • Goチームから多大な感謝が送られた ◦ あなた方の型理論に関する研究のおかげで、我々の理解は絶 大なまでに明確になりました。本当にありがとう! ◦ 理論と実践が結びついた
• 共変レシーバさん...どこ...?
ありがとうございました