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
関係性から理解する"同一性"の型用語たち
Search
Sponsored
·
SiteGround - Reliable hosting with speed, security, and support you can count on.
→
pvcresin
May 22, 2026
Programming
120
1
Share
関係性から理解する"同一性"の型用語たち
TSKaigi 2026
https://2026.tskaigi.org/talks/6
pvcresin
May 22, 2026
More Decks by pvcresin
See All by pvcresin
リセットCSSを1行消したらアクセシビリティが向上した話
pvcresin
4
520
Reduxモダナイズ 〜コードのモダン化を通して、将来のライブラリ移行に備える〜
pvcresin
2
930
text-box-trim について 1 分で
pvcresin
0
150
Web IDEの進化とそれを支える技術
pvcresin
0
130
ステップアップOSSコントリビュート
pvcresin
0
500
Webアプリケーションのアーキテクチャパターンから読み解くNext.js
pvcresin
0
440
Eight WebフロントエンドのDX向上に関する取り組み
pvcresin
0
200
TS 未経験者が 社内向け JS ライブラリを TS に置き換えている話
pvcresin
0
420
React はじめの一歩
pvcresin
1
270
Other Decks in Programming
See All in Programming
[RubyKaigi 2026] Require Hooks
palkan
1
320
Are We Really Coding 10× Faster with AI?
kohzas
0
170
要はバランスからの卒業 #yumemi_grow
kajitack
0
160
Programming with a DJ Controller — not vibe coding
m_seki
3
860
サプライチェーン攻撃対策「層を重ねて落ちない壁」を10日間で組み上げた話 #TechLeadConf2026
kashewnuts
1
280
ついに来た!本格的なマルチクラウド時代の Google Cloud
maroon1st
0
440
PicoRuby for IoT: Connecting to the Cloud with MQTT
yuuu
2
770
ふにゃっとしない名前の付け方 〜哲学で茹で上げる、コシのあるソフトウェア設計〜
shimomura
0
120
20年以上続くプロダクトでも使い続けられる静的解析ツールを求めて
matsuo_atsushi
0
150
【ディップ|26年新卒研修資料】TDD実装演習
dip_tech
PRO
0
190
When benchmarks go bad - what I learned from measuring performance wrong
hollycummins
0
390
【ディップ|26年新卒研修資料】OpenAPI/Swagger REST API研修
dip_tech
PRO
0
160
Featured
See All Featured
Reality Check: Gamification 10 Years Later
codingconduct
0
2.1k
Ten Tips & Tricks for a 🌱 transition
stuffmc
0
110
The Pragmatic Product Professional
lauravandoore
37
7.3k
The Art of Programming - Codeland 2020
erikaheidi
57
14k
Efficient Content Optimization with Google Search Console & Apps Script
katarinadahlin
PRO
1
550
Building Better People: How to give real-time feedback that sticks.
wjessup
370
20k
Agile Actions for Facilitating Distributed Teams - ADO2019
mkilby
0
190
Building AI with AI
inesmontani
PRO
1
990
4 Signs Your Business is Dying
shpigford
187
22k
Organizational Design Perspectives: An Ontology of Organizational Design Elements
kimpetersen
PRO
1
690
Skip the Path - Find Your Career Trail
mkilby
1
120
Rebuilding a faster, lazier Slack
samanthasiow
85
9.5k
Transcript
関係性から理解する "同一性"の型用語たち pvcresin TSKaigi 2026
pvcresin 仕事 SansanでEightのWebエンジニア 興味 フロントエンド/デザイン/型 言語 TypeScript/Kotlin/Ruby 自己紹介
TypeScriptの話題において、nominal / structural / branded / opaque / phantomといった 型に関する用語を見かける機会が増えてきました。
しかし、これらの違いや関係性を説明しようとすると曖昧になりがちです。 私自身も「なんとなく分かったつもり」で流してしまった過去があり、 後からうまく説明できずに後ろめたさを感じる場面がありました。 本発表ではTypeScriptが構造的型付けであることを起点に、 型の同一性に関するこれらの用語を、コード例を交えて紹介します。 そのうえで、これらの違いと関係性を理解し、解説記事や議論を追える状態を目指します。 トーク概要
この発表のスコープ 話さないこと 話すこと - コード例を交えた型用語の解説 - それぞれの用語の関係性の整理 - 特定のライブラリの使い方
型の同一性 A型の値を、 B型の値に代入できるか
アジェンダ - Nominal/Structural - プリミティブ型に関する課題 - Branded/Opaque/Phantom - まとめ
Nominal/Structural
Nominal/Structural Nominal Typing(名前的型付け) - 公称型とも呼ばれる - 名前や継承などで型を判別 - Java, C#,
Kotlin, Go(struct)など Structural Typing(構造的型付け) - 構造的部分型とも呼ばれる - 構造で型を判別 - TypeScript, Go(interface)など
Structural Typing Nominal Typing クラス
type aliasはほとんどの言語で透過的 プリミティブ
Goのdefined typeはNominal プリミティブ
トレードオフ Nominal Structural 構造が同じ値の取り違え ✅ 名前で判別できる ❌ 混ざる可能性がある 別の型として渡す ❌
変換が必要 ✅ 構造が合えば渡せる
Duck/Goose
Goose Typing - 実行時に対象の値が特定のクラスに 属しているかで型を判定 - Python(ABC + isinstance), Ruby(is_a,
kind_of, instance_of), JavaScript(instanceof)など Duck/Goose Duck Typing - 実行時にその値が特定のメソッドや プロパティを持っているかで型を判定 - コードの再利用性を高められる - Python, Ruby, JavaScriptなど
Goose Typing Duck Typing Duck/Goose ※ 呼び出し前に事前チェックする場合 ※ PythonのABC相当なし
JavaScriptでのDuck Typingを静的な型に自然に落とし込むためには、 TypeScriptはStructuralな型システムである必要があった ここまでのまとめ 名前で判別 構造で判別 コンパイル時 Nominal Structural 実行時
Goose Duck
プリミティブ型に関する課題
プリミティブ型に関する課題 - 手軽に使えるため、プリミティブな値で管理する場面は多い - ID(UserId, GroupId, …) - 単位(JPY, USD,
…) - 特定の文字列(Email, URL, …) - すべてがstringやnumberに潰れがちで、取り違えのリスクが高い - 一方、クラスは構造がユニークになることが多く、問題になりづらい
Branded
- 識別子となるプロパティを混ぜて、型を判別可能にする実装パターン Branded Type(ブランド型)
- クラス - オブジェクト - プリミティブ プリミティブな値を判別するアプローチ
- 両方とも中身が同じstringなので 判別できない - あ - あ - あ 元コード:
プリミティブな type alias
- クラスは通常Structuralに判別 - 固有の変数を持つ場合はNominal - private, protected修飾子 - JavaScriptの#private field
- 実行時オーバーヘッド クラス + private - あ - あ - あ
オブジェクトにネスト + string literal - オブジェクトにネストし、値の型を判別するための情報を追加する - 型をasで宣言することで、value以外に追加で__brandを設定しなくて済む - 実行時にオーバーヘッド
- プリミティブな値をそのまま使える - 実行時への影響がない プリミティブ + オブジェクト交差
プリミティブパターン良さそう 現状のコードだとまだ課題がある - __brandキーがstringなので、IDEの補完に出る/他コードと衝突しうる - 本当に欲しいのは 補完を汚さない/衝突しない/文字列で触れないキー
衝突しない brandプロパティを作るなら Symbol Symbol - 必ず一意になるプリミティブ(型はunique symbol) - プロパティのキーに使うことで、補完を汚さず、不正なアクセスも防ぐ
- 実行時に値を生成しないよう、Symbol()で実体を作らずにdeclare const - 衝突なし・補完に出ない・不正にアクセスができないbrandプロパティ - モジュール内では__brandに代入が可能なため、readonlyで上書き防止 キーを unique symbol
+ readonly にする
- 識別子となるプロパティを混ぜて、型を判別可能にする実装パターン 【再掲】Branded Type(ブランド型)
Opaque
Opaque Type(不透明型) - 型の実体を外部に教えず、生成口を絞る運用パターン(=カプセル化) - Scala, Swift, Flowなど
Opaque Type(不透明型) - TypeScriptには as 制限などの隠蔽機能がない - 「Branded Type +
生成口を1点に絞る運用」と捉えるのが現実的
Phantom
- 状態を表す型パラメータで判別 - 渡した型パラメータが、値の表現には 現れない - Haskell, Scala, Kotlinなど Phantom
Type(幽霊型)
代入できてしまう 素直にTypeScriptにすると
privateを導入して Nominalにすると実現可能
クラスにしないでも実現可能
まとめ
型用語の関係性 実装に使う 生成口絞る Branded Type (ブランド型) 識別子を混ぜて型を判別 Opaque Type (不透明型)
型の実体を外部に教えない Phantom Type (幽霊型) 状態を型パラメータで持つ 判定タイミング 判定方式 Structural Typing (構造的型付け) 構造で型を判別 Nominal Typing (名前的型付け) 名前で型を判別 Goose Typing 実行時に 名前で型を判別 Duck Typing 実行時に 構造で型を判別 判定方式 Nominal模倣
- TypeScriptはStructural Typingだが、型の表現力でNominalを模倣できる - Branded Typeを活用すれば、実行時への影響なく型安全性を高められる - 実装は同じ見た目でも、用語によって重視する概念が異なる まとめ
None