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
DDDについて勉強したので5分でまとめる
Search
Sponsored
·
Ship Features Fearlessly
Turn features on and off without deploys. Used by thousands of Ruby developers.
→
y_ahiru
August 28, 2018
Programming
360
0
Share
DDDについて勉強したので5分でまとめる
1ヶ月ほどDDDについて勉強したので、その成果を5分でLTしました。
y_ahiru
August 28, 2018
More Decks by y_ahiru
See All by y_ahiru
恣意性から考える、変更に強いモデルの作り方
yahiru
1
2.5k
責務と認知負荷を整える! 抽象レベルを意識した関心の分離
yahiru
10
2.9k
さいきょうのレイヤードアーキテクチャについて考えてみた
yahiru
3
1.1k
フロントエンドエンジニアも知っておきたい HTTP/3 で変わること
yahiru
16
13k
ゆるふわCQRS入門
yahiru
2
780
設計におけるソリューションドメイン
yahiru
3
1.8k
PHPで始めるGitHub Actions
yahiru
1
870
5ヶ月でカバレッジを20%から90%にあげた話
yahiru
2
7k
入門ミューテーションテスト/ A bigginer's guide to Mutation testing
yahiru
0
1.6k
Other Decks in Programming
See All in Programming
モダンOBSプラグイン開発
umireon
0
190
「接続」—パフォーマンスチューニングの最後の一手 〜点と点を結ぶ、その一瞬のために〜
kentaroutakeda
5
2.3k
コードレビューをしない選択 #でぃーぷらすトウキョウ
kajitack
3
1.2k
モックわからないマン卒業記 ~振る舞いを起点に見直した、フロントエンドテストにおけるモックの使いどころ~
tasukuwatanabe
3
440
Symfony + NelmioApiDocBundle を使った スキーマ駆動開発 / Schema Driven Development with NelmioApiDocBundle
okashoi
0
250
AIと共にエンジニアとPMの “二刀流”を実現する
naruogram
0
110
How to stabilize UI tests using XCTest
akkeylab
0
150
[PHPerKaigi 2026]PHPerKaigi2025の企画CodeGolfが最高すぎて社内で内製して半年運営して得た内製と運営の知見
ikezoemakoto
0
310
「効かない!」依存性注入(DI)を活用したAPI Platformのエラーハンドリング奮闘記
mkmk884
0
280
PHPのバージョンアップ時にも役立ったAST(2026年版)
matsuo_atsushi
0
280
Geminiをパートナーに神社DXシステムを個人開発した話(いなめぐDX 開発振り返り)
fujiba
0
130
LM Linkで(非力な!)ノートPCでローカルLLM
seosoft
0
300
Featured
See All Featured
A brief & incomplete history of UX Design for the World Wide Web: 1989–2019
jct
1
340
From π to Pie charts
rasagy
0
160
Jamie Indigo - Trashchat’s Guide to Black Boxes: Technical SEO Tactics for LLMs
techseoconnect
PRO
0
92
Dominate Local Search Results - an insider guide to GBP, reviews, and Local SEO
greggifford
PRO
0
120
Mind Mapping
helmedeiros
PRO
1
140
Darren the Foodie - Storyboard
khoart
PRO
3
3.1k
No one is an island. Learnings from fostering a developers community.
thoeni
21
3.7k
Improving Core Web Vitals using Speculation Rules API
sergeychernyshev
21
1.4k
Claude Code どこまでも/ Claude Code Everywhere
nwiizo
64
54k
Connecting the Dots Between Site Speed, User Experience & Your Business [WebExpo 2025]
tammyeverts
11
870
Why Your Marketing Sucks and What You Can Do About It - Sophie Logan
marketingsoph
0
120
Understanding Cognitive Biases in Performance Measurement
bluesmoon
32
2.8k
Transcript
DDDの勉強をしたので5分でまとめる
自己紹介 吉田です Twitter @strtyuu
ドメイン駆動設計とは? ドメインを モデリングして モデリングして モデリングし続ける設計手法のこと
ドメインとは? アプリケーションが解決したい問題そのもののこと
モデリングとは? モデルの振る舞いを定義していく作業のことを指します。
モデルとは? ドメインを構成する様々な要素のこと
たとえば 会議室の予約システムを作りますとなった場合 会議室はもちろんモデルになると思いますし 予約という概念もモデルになります。
モデリングのやり方 ドメインエキスパートとドメインについて話し合って 話し合いを設計に反映させて 設計を元に実装をして ドメインエキスパートとドメインについて話し合って 話し合いを設計に反映させて 設計を元に実装をして ドメインエキスパートと...
以下エンドレス
モデルがどんどん更新されてゆく 実装は更新されてゆくモデルについていかなければならない つまり、変更に対して柔軟な実装になっていないと、 ドメイン駆動設計っていうのは、実現できない
柔軟な実装をするために デザインパターンを駆使して、複雑さに立ち向かう Repository Entity ValueObject Layered Architecture Aggregate 他
レイヤードアーキテクチャ アプリケーションをいくつかの層に分けて それぞれの役割を明確に区別すること
レイヤー化するメリット 層同士の依存関係が最低限に抑えられるので 層のインターフェースが変わらなければ、他の層への影響はあまりない (ことが多い) つまり、変更がしやすくなる。
たとえば <?php class ApplicationLayer { public function reserve(Reservation $reservation) {
if ( ! $reservation->isReservable()) { // 予約不可な場合の処理 } // 登録処理 } } 予約が可能かどうかのロジックがいくら変わろうが、アプリケーション 層は変更せずにそのまま使用できる
基本的な4つの層 プレゼンテーション層 アプリケーション層 ドメイン層 インフラ層
プレゼンテーション層 アプリケーションの外側と内側の接点 最終的にアプリケーションの外側に渡したいデータを出力する役割を持 っている
アプリケーション層 ドメインロジック以外の処理を担当する
ドメイン層 ドメインのルール・ロジックを全て担当する。 ドメイン層以外はドメインロジックを持ってはいけない
インフラ層 データベースとのやりとりなどを担当する
ドメイン層を構成する要素 Domain Service Entity ValueObject Factory
ValueObject イミュータブル(不変)なオブジェクト 値としてあるべき姿や振る舞いを定義してあげる
たとえば 会議室の予約に通常の予約の他に「仮予約」や「キャンセル」などのス テータスがあった場合、予約のステータスの値には特定の値しか存在し てほしくありません。 そういったときなどにValueObjectを作ります。
Entity ミュータブル(可変)なオブジェクト 1つの1レコードをオブジェクト化するイメージ 複数のValueObjectやリレーション先のEntityなどで構成される 様々なドメインルールを表現する
たとえば キャンセルされた予約は予約済みに戻すことができないというドメイン ルールがあった場合、以下のような感じで表現出来ます <?php class Reservation { protected $status; function
reservedBy(User $user) { if ($this->status->isCancel()) { throw new LogicException(); } $this->user_id = $user->getUserId(); $this->status = new Status('Reserved'); } }
たとえば こういった書き方をすることによって、ドメインルールと矛盾した処理 をアプリケーション層で書くことが不可能になります <?php class ApplicationLayer { public function badReserve()
{ $canceledRsv = $this->rsv_repo->findById($id); $user = $this->user_repo->findById($user_id); // 絶対に例外が発生する $canceledRsv->reservedBy($user); $this->rsv_repo->save($reservation); } }
Factory Entityを生成する際に複雑な処理が必要な時などに使う
Domain Service Entity単体では対応しきれないドメインルールを担当する
たとえば ユーザーのメールアドレスは重複してはならないというドメインルール があった場合 これは複数のEntityに関わることなので1つのEntityだけでは判断しきれ ません。 そういった処理はDomain Serviceの仕事となります。
インフラ層を構成する要素 Aggregate Repository
Aggregate ライフサイクルが同じオブジェクトをひとまとめにする「概念」
たとえば 会議室の予約に、利用者一覧みたいなデータを登録できるとします。 利用者の人数は可変なので予約とは別テーブルにすることにします。 テーブルは別ですが、新規予約時に予約と一緒に利用者も登録されます し、予約の編集時に利用者も編集されますし、予約の削除時に利用者も 削除されるかと思います。 また、予約をどうこうする時以外は利用者のデータは変更されなさそう です。 こういった場合は予約と利用者は一つの集約と言えます。
Repository データベースへのアクセスを提供する役割を持つ。 Aggregateごとに作成して、整合性を保つ。 AggregateのルートになるEntity以外へは直接アクセスさせない。
たとえば 先ほどの例でいうと 予約がAggregateのルートになり、予約Entityからしか利用者のデータに は辿れなくします。 そうすることによって、Aggregate内の整合性を保ちやすくします。
たとえば ルートとなるReservationに利用者は10人までしか登録できないというド メインルールを定義します。 <?php class Reservation { protected $users =
[]; public function addUser(User $user) { if (count($this->users) > 10) { throw new LogicException(); } $this->users[] = $user; } }
たとえば そうすると、もちろんReservationEntityを使う限り10人を超えた利用者 は登録できなくなります <?php class ApplicationLayer { public function badAddUser()
{ // たくさんのユーザー $users = $this->user_repo->getAllUsers(); $reservation = $this->rsv_repo->findById($id); foreach ($users as $user) { // 10 人を越えると必ず例外が発生する $reservation->addUser($user); } $this->rsv_repo->save($reservation); } }
たとえば そして、RepositoryはルートEntityしか受け付けないようにすることで ドメインルールに適合したデータのみがDBへ書き込まれることが保証さ れます。 <?php class ReservationRepository { public function
save(Reservation $reservation) { // DB への書き込み処理 } }
まとめ デザインパターンを駆使して、変更が容易な設計にしよう! 変更を容易にしてモデリングをしまくろう! 良いアプリケーション、作ろうぜ!