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
·
Your Podcast. Everywhere. Effortlessly.
Share. Educate. Inspire. Entertain. You do you. We'll handle the rest.
→
hakobe (Yohei Fushii)
August 25, 2016
Programming
21k
6
Share
Embed
Copy iframe code
Copy JS code
Copy link
Start on current slide
契約による設計の紹介
hakobe (Yohei Fushii)
August 25, 2016
More Decks by hakobe (Yohei Fushii)
See All by hakobe (Yohei Fushii)
はてなにおけるプログラミング言語の歴史とこれから
hakobe
0
790
ざっとわかるPython
hakobe
4
7k
はてなで一人前のエンジニアになる方法
hakobe
3
7k
アクターによる 並行処理アプケーション アーキテクチャ
hakobe
4
3.6k
Scala In Perl Company
hakobe
11
12k
TDDの練習 Coding Kata の実践
hakobe
0
1.4k
なめらかにGHEに移行する方法
hakobe
19
7.9k
開発フロー@はてなブックマーク
hakobe
0
190
Git on WebApp with Perl
hakobe
1
3.9k
Other Decks in Programming
See All in Programming
AI 時代のソフトウェア設計の学び方
masuda220
PRO
29
12k
Webフレームワークの ベンチマークについて
yusukebe
0
160
軽量Java基盤の設計 DIコンテナに頼らない、長期保守と1秒起動の実現 JJUG CCC 2026 Spring
macha64
0
490
生成AI時代にこそ効くGo | Why Go Works in the Age of Generative AI
mom0tomo
8
3.2k
スマートグラスで並列バイブコーディング
hyshu
0
120
AIとASP.NET Coreで雑Webアプリを作った話
mayuki
0
500
Oxcを導入して開発体験が向上した話
yug1224
4
310
Swiftのレキシカルスコープ管理
kntkymt
0
220
Skillsは効率化、Agentsは"自分の拡張"——Builder時代のエージェント編成(CC Night 2026)
wemra
1
120
dRuby over BLE
makicamel
2
330
「エンジニアインターン、どうやって取った?」準備のリアルを語るLT会 Progate BAR
akiomatic
0
130
jQueryをバージョンアップする前に使いたいjQuery Migrate
matsuo_atsushi
0
200
Featured
See All Featured
Dealing with People You Can't Stand - Big Design 2015
cassininazir
367
27k
Heart Work Chapter 1 - Part 1
lfama
PRO
7
36k
The B2B funnel & how to create a winning content strategy
katarinadahlin
PRO
1
380
The Power of CSS Pseudo Elements
geoffreycrofte
82
6.3k
Building Experiences: Design Systems, User Experience, and Full Site Editing
marktimemedia
0
530
The Invisible Side of Design
smashingmag
302
52k
Hiding What from Whom? A Critical Review of the History of Programming languages for Music
tomoyanonymous
2
850
Color Theory Basics | Prateek | Gurzu
gurzu
0
360
The Curious Case for Waylosing
cassininazir
1
380
WENDY [Excerpt]
tessaabrams
11
38k
Bash Introduction
62gerente
615
220k
Ethics towards AI in product and experience design
skipperchong
2
310
Transcript
ܖʹΑΔઃܭͷհ hakobe932
: ؔͷೖྗग़ྗͷ νΣοΫͲ͏ͯ͠·͔͢?
ؔͷ༷: τʔΫϯΛड͚औΓHTTPϦΫΤετͯ͠ ಘΒΕͨ༰Λฦ͢ sub retrieve_content_by_token_1 { my ($token) =
@_; my $url = 'http://www.example.com/api?token=' . $token; my $res = request($url); return $res->content; } ↑ ԿͷΤϥʔνΣοΫͳ͍ ۭจࣈྻͩͬͨΒ ϦΫΤετࣦഊͯͨ͠Β
sub retrieve_content_by_token_return { my ($token) = @_; return unless $token;
my $url = 'http://www.example.com/api?token=' . $token; my $res = request($url); return if $res->is_error; return $res->content; } Կ͔͓͔͠ͳ͜ͱ͕ىͬͨ͜ΒِʹͳΔΛฦ͢ ͓͔͔ͬͨ͠ΒSFUVSO ͬͪ͜ ྑ͘ͳ͍͜ͱ͕ىͬͨ͜Βِ
sub retrieve_content_by_token_3 { my ($token) = @_; $token //= 'DEFAULT_TOKEN';
my $url = 'http://www.example.com/api?token=' . $token; my $res = request($url); return Failure->new($res->code) if $res->is_error; return Success->new($res->content); } ͱʹ͔͘ݺͼग़ͤΔ/݁Ռͷදݱ2छྨ ࣦഊͨ͠Β'BJMVSFΦϒδΣΫτ ޭͨ͠Β4VDDFTTΦϒδΣΫτ ͳΔ͘Ժศʹࡁ·͢ ೖྗ͕͓͔͔ͬͨ͠ΒσϑΥϧτ
sub retrieve_content_by_token_die { my ($token) = @_; die 'no token'
unless $token; my $url = 'http://www.example.com/api?token=' . $token; my $res = request($url); die 'request error: ' . $res->content if $res->is_error; return $res->content; } ॲཧ͕ࣦഊͨ͠Βྫ֎Λൃੜͤ͞Δ ෆਖ਼ͳೖྗڐ͞ͳ͍ ྫ֎ͰͳΜͱ͔͢Δ
Ͳͷೖྗݕࠪॲཧ͕ྑ͍Ͱ͠ΐ͏? ❖ ྑ͘ͳ͍͜ͱ͕ىͬͨ͜Βِ ❖ ͳΔ͘Ժศʹࡁ·͢ ❖ ྫ֎ͰͳΜͱ͔͢Δ
ਖ਼ղ: Θ͖ͨ͠ʹͨ͠ ͲͷΑ͏ʹߟ͑Δ͖͔ͷࢦ͋Δ ܅Β͖ʹ͠Ζ
ߟ͑Δ͖͜ͱ: ؔͷ༷
έʔεελσΟʹ͓͚Δؔͷ༷ •τʔΫϯΛड͚औΓ •HTTPϦΫΤετͯ͠ •ಘΒΕͨ༰Λฦ͢ ←ਖ਼͍͠ೖྗ͕ಘΒΕͨ࣌ ←Կ͔ࣄΛͯ͠ ←ظ͞ΕΔ݁ՌΛฦ͢
έʔεελσΟʹ͓͚Δؔͷ༷ (Cont. •τʔΫϯΛड͚औΓ •HTTPϦΫΤετͯ͠ •ಘΒΕͨ༰Λฦ͢ ←ࣄલ݅Λຬͨͤ ←Կ͔ࣄΛͯ͠ ←ࣄޙ݅Λຬͨ͢ ؔͷར༻ऀ͕ ؔ
Q: ͕݅͠ຬͨͤͳ͍ͱͲ͏ͳΔ? ❖ ࣄલ͕݅ຬͨͤͳ͍ • ex.τʔΫϯ͕͞Εͳ͔ͬͨ • HTTPϦΫΤετΛ͢ΔͨΊͷURL͕ߏஙͰ͖ͳ͍ ❖ ࣄޙ͕݅ຬͨͤͳ͍
• ex. HTTPϦΫΤετ͕ࣦഊͨ͠ • ฦ͖͢ίϯςϯπ͕࡞Εͳ͍ A: ؔతΛՌͨͤͳ͍ ؔΛ࣮ߦͨ݁͠ՌɺԿ͕ى͜Δ͔ෆఆ
ࣄલ݅ͱࣄޙ݅ͷද໌ ❖ ؔΛਖ਼͘͠ಈ͘ͷʹඞཁͳ݅Λද໌͢Δ # 事前条件: $token は必ず渡される必要がある # 事後条件: 取得したコンテンツは必ず返却される
sub retrieve_content { my ($token) = @_; my $url = 'http://www.example.com/api?token=' . $token; my $res = request($url); return $res->content; }
use Carp::Assert; sub retrieve_content_assert { my ($token) = @_; assert($token);
my $url = 'http://www.example.com/api?token=' . $token; my $res = request($url); assert($res->is_success && $res->content); return $res->content; } ίʔυͰදݱ͢Δͱ࣮ߦ࣌ʹνΣοΫͰ͖Δ ίʔυʹΑΔද໌ͷදݱ(Carp::Assert) τʔΫϯΛ͏͚ͱ͍ͬͯΔ͜ͱͷද໌ ίϯςϯπΛ༻ҙͰ͖ͨ͜ͱͷද໌
sub retrieve_content_by_token_die { my ($token) = @_; die 'no token'
unless $token; my $url = 'http://www.example.com/api?token=' . $token; my $res = request($url); die 'request error: ' . $res->content if $res->is_error; return $res->content; } PerlͰૉʹྫ֎ʹΑΓදݱ͢Δ͘Β͍͕Ұൠత ίʔυʹΑΔද໌ͷදݱ(ྫ֎ ) τʔΫϯΛ͏͚ͱ͍ͬͯΔ͜ͱͷද໌ ίϯςϯπΛ༻ҙͰ͖ͨ͜ͱͷද໌
sub retrieve_content_by_token_die { my ($token) = @_; die 'no token'
unless $token; my $url = 'http://www.example.com/api?token=' . $token; my $res = request($url); die 'request error: ' . $res->content if $res->is_error; return $res->content; } ؔͷ༷ʹج͍࣮ͮͨ τʔΫϯΛड͚औΓHTTPϦΫΤετͯ͠ಘΒΕͨ ༰Λฦ͢
ܖʹΑΔઃܭ (Design by Contract) ❖ ࣄલ݅ͱࣄޙ݅ͱ͍͏એݴతͳੑ࣭ʹج͍ͮͯ ؔΛઃܭ͢Δ ❖ ؔͱͦͷར༻ऀͷؒʹˢͷΑ͏ͳܖ͕͋Δͱ ଊ͑Δ
ʮͦͪ͠Β͕ࣄલ݅Λຬͨͨ͠ঢ়ଶͰࢲΛ ݺͿͱଋͯ͠Լ͞ΔͳΒɺ ͓ฦ͠ʹࣄޙ݅Λຬͨ͢ঢ়ଶΛ࠷ऴతʹ࣮ݱ ͢Δ͜ͱΛ͓ଋ͠·͢ʯ
ܖʹΑΔઃܭͷར ❖ ਖ਼͍͠ίʔυΛॻ͘ॿ͚ʹͳΔ • ਖ਼͍ؔ͠ͷ͍ํ͕໌֬ʹͳΔ • ·͕͍ͪʹૣ͘ؾ͚ͮΔ • → ιϑτΣΞͷ৴པੑ্͕͢Δ
❖ ؔͷੑ࣭Λੳ͢Δνϟϯε͕ಘΒΕΔ ❖ ؔͷੑ࣭Λදݱ͢ΔυΩϡϝϯτʹͳΔ
Q:ΤϥʔͳΜͰྫ֎ʹ͢Δ? ❖ ͏͢͜͠ৄ͘͠ • ؔͷ༷Λਖ਼͘͠ఆٛ͠Α͏ • ཁٻܕͷؔ/อޢܕͷؔ • ৴པͰ͖ͳ͍ೖྗ༧ΊνΣοΫ͠Α͏ A:
ͦ͏Ͱͳ͍
sub retrieve_content_by_token_die { my ($token) = @_; die 'no token'
unless $token; my $url = 'http://www.example.com/api?token=' . $token; my $res = request($url); die 'request error: ' . $res->content if $res->is_error; return $res->content; } ؔͷ༷ʹج͍࣮ͮͨ(࠶ܝ) τʔΫϯΛड͚औΓHTTPϦΫΤετͯ͠ಘΒΕͨ ༰Λฦ͢ Τϥʔͷ༰ʹԠͯ͡ ॲཧΛΓସ͍͑ͨ
ؔͷ༷Λదʹఆٛ͠Α͏ τʔΫϯΛड͚औΓHTTPϦΫΤετͯ͠ಘΒΕͨ ༰Λฦ͢ τʔΫϯΛड͚औΓHTTPϦΫΤετͯ͠ ޭ͢ΕಘΒΕͨ༰Λฦ͢ ࣦഊ͢ΕΤϥʔͷछྨΛฦ͢ ؔͱͷదͳܖؔΛߟ͑Δ → ઃܭ
sub retrieve_content_by_token_2 { my ($token) = @_; die 'no token'
unless $token; my $url = 'http://www.example.com/api?token=' . $token; my $res = request($url); return Failure->new($res->code) if $res->is_error; return Success->new($res->content); } ࣄલ݅Λຬͨͯ͠ͳ͍ͷͰྫ֎ ޭͨ͠Β4VDDFTTΦϒδΣΫτ ৽͍ؔ͠ͷ༷ʹج͍࣮ͨ ࣦഊͨ͠Β'BJMVSFΦϒδΣΫτ τʔΫϯΛड͚औΓ HTTPϦΫΤετͯ͠ ޭ͢ΕಘΒΕͨ༰Λฦ͢/ࣦഊ͢ΕΤϥʔͷछྨΛฦ͢
sub retrieve_content_by_token_3 { my ($token) = @_; $token //= 'DEFAULT_TOKEN';
my $url = 'http://www.example.com/api?token=' . $token; my $res = request($url); return Failure->new($res->code) if $res->is_error; return Success->new($res->content); } τʔΫϯͳ͚ΕσϑΥϧτ ޭͨ͠Β4VDDFTTΦϒδΣΫτ ͞Βʹࣄલ݅ΛΏΔΊΔ͜ͱͰ͖Δ ࣦഊͨ͠Β'BJMVSFΦϒδΣΫτ τʔΫϯΛड͚औΔ͕ɺͳ͚ΕσϑΥϧτΛ͍ HTTPϦΫΤετͯ͠ ޭ͢ΕಘΒΕͨ༰Λฦ͢/ࣦഊ͢ΕΤϥʔͷछྨΛฦ͢
ཁٻܕͷؔͱอޢܕͷؔ ❖ ཁٻܕͷؔ • ݺͼग़͠ଆ͕ࣄલ݅Λຬͨ͢͜ͱΛཁٻ͢Δؔ • ex. ͯ͢ͷॴͰྫ֎Λ͍ͬͯͨύλʔϯ ❖ อޢܕͷؔ
• ͲΜͳೖྗͰͳΜͱͯ͠ॲཧ͢Δؔ • ex. Ұ൪݅ΛΏΔΊͨύλʔϯ ͲͪΒ͕ྑ͍͔ɺ࣌ͱ߹ɺΈʹΑΔ
ཁٻܕ͕͓͢͢Ί(υϝΠϯϩδοΫͰ) ❖ ͏ଆʹڧ͍݅Λ՝͢͜ͱͰ͕ؔਖ਼͘͠ಈ࡞͢Δ ͜ͱΛڧ੍Ͱ͖Δ ❖ ҟৗͳೖྗΛॲཧ͢Δίʔυ͕ͳ͘ͳΓίΞϩδοΫ ʹूதͨ͠γϯϓϧͳίʔυʹͳΔ ❖ ҟৗͳೖྗΛͲͷΑ͏ʹ੍ޚ͢Δ͔͏ଆͷΈͰ ܾΊΔͷͰͷॴࡏ͕໌Β͔
sub retrieve_content_by_token_3 { my ($token) = @_; $token //= 'DEFAULT_TOKEN';
my $url = 'http://www.example.com/api?token=' . $token; my $res = request($url); return Failure->new($res->code) if $res->is_error; return Success->new($res->content); } ҾΛ͠Θ͢Εͯͯಈ͘ อޢܕͷؔͷ߹ ຊ࣭తͰͳ͍ॲཧ͕૿͑Δ ೖྗݕࠪͷ͕ᐆດʹ
sub retrieve_content_by_token_die { my ($token) = @_; die 'no token'
unless $token; my $url = 'http://www.example.com/api?token=' . $token; my $res = request($url); die 'request error: ' . $res->content if $res->is_error; return $res->content; } ཁٻܕͷؔͷ߹ ෆਖ਼ͳݺͼग़͠Τϥʔ ༷ʹԊ͏ݕࠪͷΈΛߦ͏ ೖྗݕࠪͷݺͼग़͠ଆ
my $input_token = <STDIN>; if (!$input_token) { print "Please input
token"; exit 1; } my $content = retrieve_content_by_token_die($token); ࣄલ݅ΛνΣοΫͯ͠ݺͼग़͢ ࣄલ݅Λຬͨ͢͜ͱͷνΣοΫ ҆৺ͯ͠ݺͼग़͢ Ϟσϧͷ ϩδοΫ Ϣʔβೖྗ ϑΟϧλ
ؔͷ༷Ͳ͏ܾͬͯΊΔͷ͔? ❖ ؔʹͲͷΑ͏ͳΛ༩͑Δͷ͔ ؔͷଐ͢ΔΫϥεϞδϡʔϧʹґଘ͢Δ ❖ ΫϥεϞδϡʔϧͷ ιϑτΣΞͷΞʔΩςΫνϟʹґଘ͢Δ ࡉ෦ͷઃܭશମͷઃܭʹجͮ͘
ϨΠϠͷ͋ΔΞʔΩςΫνϟͷ߹ Ϣʔβೖྗ • ೖྗͷνΣοΫ • νΣοΫOK → ϩδοΫݺͿ • νΣοΫNG
→ ΤϥʔΛฦ͢ • ೖྗΛ৴པͯ͠ υϝΠϯϩδοΫ Λ࣮ߦ อޢܕͷϝιου ཁٻܕͷϝιου ΠϯλʔϑΣʔε υϝΠϯϩδοΫ ݺͼग़͠ ϨΠϠͷʹରԠ͢Δ
ܖʹΑΔઃܭؔʹݶΒͳ͍ ❖ ؔදత͚ͩͲଞͷ෦ʹద༻Ͱ͖Δ • Ϋϥε: ΦϒδΣΫτͷෆม݅ • ϧʔϓ: ϧʔϓෆม݅ ίϯϙʔωϯτͷཁٻΛຬͨ͢͜ͱͰརӹΛಘΔ
ৄ͘͠ΦϒδΣΫτࢦೖΛಡ͏ ❖ ιϑτΣΞઃܭͷຊ ❖ ιϑτΣΞ࣭ͷఆٛ ྺ࢙ΛͨͲֶͬͯΔ ❖ ༗ӹ͚ͩͲྔ͕͓͓͍ • 1800ϖʔδ͘Β͍
• ͕ΜΖ͏
ιϑτΣΞΞʔΩςΫνϟ͕ؾʹͳΔਓ͚ https://8thlight.com/blog/uncle-bob/2012/08/13/the-clean-architecture.html Clean Architecture
·ͱΊ ❖ ʮܖʹΑΔઃܭʯਖ਼͍͠ίʔυͷͨΊͷςΫχοΫ • ؔͷ༷Λࣄલ݅ͱࣄޙ͔݅ΒͳΔܖͱଊ͑Α͏ • ܖʹج͍ͮͯίʔυΛ࣮͠Α͏ ❖ ؔͱͷܖιϑτΣΞΞʔΩςΫνϟΛݩʹ ߟ͑Α͏
• ΫϥεϨΠϠͷΛੳͯ͠ରԠͨ͠Λ༩͑Α͏ • ཁٻܕͱอޢܕΛҙ͍ࣝͯ͠Θ͚Α͏
ޚతϓϩάϥϛϯάͲ͜? ❖ Կ͕ى͜Δ͔Θ͔Βͳ͍ͷͰݸʑͷؔͳΔ͘ ͋ΒΏΔೖྗνΣοΫ ❖ ؔͷఆٛͱݺͼग़͠ܖͱ͍͏ܗͰཧ͞Ε͍ͯ Εγϯϓϧͳίʔυͱ໌͕֬͞ҡ࣋͞ΕΔ