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
Pythonソースコードの構造可視化とそれがもたらすもの
Search
yosu
August 28, 2020
Programming
5.6k
5
Share
Embed
Copy iframe code
Copy JS code
Copy link
Start on current slide
Pythonソースコードの構造可視化とそれがもたらすもの
yosu
August 28, 2020
Other Decks in Programming
See All in Programming
The ROI of Quarkus for Spring Boot Applications
hollycummins
0
120
Strategic Design in the Frontend: Moduliths & Micro Frontends @DDDEurope
manfredsteyer
PRO
0
100
「なぜそう決めたのか」を残し続ける仕組み ― Notion AI カスタムエージェント × Slack連携による設計判断の自動記録 - NIKKEI Tech Talk #47
niftycorp
PRO
0
170
[2026年度第1回ORセミナー] 計画最適化ベンチャーと競技プログラミング人材
terryu16
0
260
net-httpのHTTP/2対応について
naruse
0
490
Language Server 使ってる? 〜VSCode と Zed の場合〜 / Are you using a Language Server? ~For VS Code and Zed~
handlename
0
790
気づいたらRubyで100作品 ー クリエイティブコーディングが生活の一部になるまで / 100 Ruby Sketches Later: How Creative Coding Became Part of My Life
chobishiba
3
580
メソッドのジェネリクスでGoの夢は広がるか? / Kyoto.go #65
utgwkk
3
780
例外の正しい扱い方 そのエラー try-catchして大丈夫?
jinwatanabe
0
240
Even G2とAWSで推しのエージェントを召喚しよう!
har1101
1
120
その問い、本当に正しいですか?AI時代のエンジニアに必要な哲学と認知科学 / ai-philosophy-cognitive-science
minodriven
9
5.2k
Spec Driven Development | AI Summit Lisbon
danielsogl
PRO
0
190
Featured
See All Featured
Agile Actions for Facilitating Distributed Teams - ADO2019
mkilby
0
210
Getting science done with accelerated Python computing platforms
jacobtomlinson
2
230
Money Talks: Using Revenue to Get Sh*t Done
nikkihalliwell
0
250
Lightning Talk: Beautiful Slides for Beginners
inesmontani
PRO
2
580
ReactJS: Keep Simple. Everything can be a component!
pedronauck
666
130k
The Art of Programming - Codeland 2020
erikaheidi
57
14k
Everyday Curiosity
cassininazir
0
230
So, you think you're a good person
axbom
PRO
2
2.1k
Why Mistakes Are the Best Teachers: Turning Failure into a Pathway for Growth
auna
0
160
Paper Plane (Part 1)
katiecoart
PRO
0
9k
My Coaching Mixtape
mlcsv
0
150
Automating Front-end Workflow
addyosmani
1370
210k
Transcript
Pythonソースコードの 構造可視化 とそれがもたらすもの PyCon JP 2020 #pyconjp_2
自己紹介 @yosu 2 years Web App & Crawler 今月まで @
お手伝い @
本日お話すること 背景 見えてくるもの 構造 と その重要性 可視化の 必要性
背景 見えてくるもの 構造 と その重要性 可視化の 必要性 本日お話すること
背景 500ファイル以上あるPythonアプリケーションのPython2/3移行プロジェクト に参加 最初に移行するべき対象を見つけ出すため、簡易的なスクリプトで各パッケージ 間の依存関係を明らかにした。 うまく対象を選び、最初の移行は大きなトラブルもなく無事完了。 他にも、実際に可視化して見ると循環参照のような望ましくない構造や、依存が 多すぎるモジュールなど改善に生かす手がかりが得られることに気づいた。
背景 見えてくるもの 構造 と その重要性 可視化の 必要性
構造とは?
None
一つの全体を構成する 部分同士の関係性 部分はそれぞれが 役割・責務を持つ
None
屋根 窓 床 天井 階段
屋根 窓 床 階段 天井 雨を防ぐ
屋根 窓 床 階段 天井 光を通す 雨を防ぐ
屋根 窓 床 階段 天井 光を通す 雨を防ぐ 1階と2階 をつなぐ
Python アプリケーション の場合
None
None
PI = 3.1415926535
PI = 3.1415926535 def area(radius): return PI * radius **
2
PI = 3.1415926535 class Circle: def __init__(self, radius): self.radius =
radius def area(self): return PI * self.radius ** 2
import sys from figure import Circle def main(): radius =
float(sys.argv[1]) circle = Circle(radius) print(circle.area())
area() PI main() Circle figure 参照 参照 参照 所有 所有
呼出し
area() PI main() Circle figure 依存 依存 依存 依存 依存
依存
area() PI main() Circle figure 名前が 役割・責務 を表す
依存関係の 特徴
A B 依存先 がないと機能しない ❌
A B 依存先 の変化で影響を受ける
今回のスコープ
全体を捉えるために、 モジュール同士の 依存関係をみていく
jig-py ⚒
github.com/levii/jig-py
イメージ
jig-pyの由来 Java製のコードによる設計支援ツール Jig(irofさん作)に由来。 https://github.com/dddjava/jig 三層+ドメインモデルの アーキテクチャを想定し、 様々な可視化ができる。
jig-pyはまだ モジュール関係のみ 可視化
話を戻して ↩
なぜ構造に 着目するか?
ソフトウェア だから
ソフト =柔軟に変化できる ここに価値がある
None
つまりソフトウェアの優位性は、開発サイクルが短く、 アイデアを出してからプロトタイプを作り、エラーを 発見して修正するまでが短時間かつローコストでできる ことにある。 もし設計プロセスで技術者が絶対にミスを犯さないのな ら、ハードウェアもソフトウェアもコストはさほど変わ らないかもしれない。だが実際にはミスは避けられない のだから、何か特別な理由でもない限り、ソフトウェア のほうが好ましいことになる。 ソフトウェアの優位性
つまりソフトウェアの優位性は、開発サイクルが短く、 アイデアを出してからプロトタイプを作り、エラーを 発見して修正するまでが短時間かつローコストでできる ことにある。 もし設計プロセスで技術者が絶対にミスを犯さないのな ら、ハードウェアもソフトウェアもコストはさほど変わ らないかもしれない。だが実際にはミスは避けられない のだから、何か特別な理由でもない限り、ソフトウェア のほうが好ましいことになる。 ソフトウェアの優位性
構造が影響を及ぼす
None
リファクタリングとは ソフトウェアの 外部的振る舞いを保ったままで、 内部の構造を改善していく作業のこと。 リファクタリングを行なえば、以前に書いたコードの 設計が向上することになる。
わるい構造
よい構造 わるい構造
よい構造 わるい構造 リファクタリング
よい構造 リファクタリング 変更が危険で やっかい 機能追加・バグフィクス が容易 開発スピードが安定 わるい構造
よい構造 リファクタリング 変更が危険で やっかい 機能追加・バグフィクス が容易 開発スピードが安定 わるい構造
どうやって 構造を みていくか
None
None
None
そのままでは 見えにくい
背景 見えてくるもの 構造 と その重要性 可視化の 必要性
None
単純に可視化 してみると
None
None
うまく可視化する必要
Google Earth
None
None
None
Python アプリケーション を探索
None
None
None
None
None
None
None
None
None
None
を使って 探索的に可視化する
① 初期状態
ステップ1: 解析・情報収集 "common.bind" -> "common.group.application"; "common.context.builder" -> "common.context.domain"; "event.application.event_history" ->
"framework.paginator"; "event.application.repository" -> "event.domain"; "feedback.application.announcement" -> "common.exceptions"; "feedback.application.idea" -> "framework.logging"; "feedback.domain.idea" -> "event.domain"; "feedback.domain.idea" -> "notify.domain.email_builder"; "framework.paginator" -> "framework.logging"; "integration.bind" -> "event.domain.event_handler_setting"; "main" -> "framework"; "main" -> "task.routes"; "notify.application" -> "common.user.application"; "task.domain.notification" -> "common.system_setting.application"; "test_helpers.feedback" -> "common.user.domain";
ステップ2: トップレベルに集約 "common.bind" -> "common.group.application"; "common.context.builder" -> "common.context.domain"; "event.application.event_history" ->
"framework.paginator"; "event.application.repository" -> "event.domain"; "feedback.application.announcement" -> "common.exceptions"; "feedback.application.idea" -> "framework.logging"; "feedback.domain.idea" -> "event.domain"; "feedback.domain.idea" -> "notify.domain.email_builder"; "framework.paginator" -> "framework.logging"; "integration.bind" -> "event.domain.event_handler_setting"; "main" -> "framework"; "main" -> "task.routes"; "notify.application" -> "common.user.application"; "task.domain.notification" -> "common.system_setting.application"; "test_helpers.feedback" -> "common.user.domain";
ステップ2: トップレベルに集約 "common" -> "common"; "common" -> "common"; "event" ->
"framework"; "event" -> "event"; "feedback" -> "common"; "feedback" -> "framework"; "feedback" -> "event"; "feedback" -> "notify"; "framework" -> "framework"; "integration" -> "event"; "main" -> "framework"; "main" -> "task"; "notify" -> "common"; "task" -> "common"; "test_helpers" -> "common";
ステップ3: 重複の除去 "common" -> "common"; "common" -> "common"; "event" ->
"framework"; "event" -> "event"; "feedback" -> "common"; "feedback" -> "framework"; "feedback" -> "event"; "feedback" -> "notify"; "framework" -> "framework"; "integration" -> "event"; "main" -> "framework"; "main" -> "task"; "notify" -> "common"; "task" -> "common"; "test_helpers" -> "common";
"common" -> "common"; "event" -> "framework"; "event" -> "event"; "feedback"
-> "common"; "feedback" -> "framework"; "feedback" -> "event"; "feedback" -> "notify"; "framework" -> "framework"; "integration" -> "event"; "main" -> "framework"; "main" -> "task"; "notify" -> "common"; "task" -> "common"; "test_helpers" -> "common"; ステップ3: 重複の除去
ステップ4: セルフループの削除 "common" -> "common"; "event" -> "framework"; "event" ->
"event"; "feedback" -> "common"; "feedback" -> "framework"; "feedback" -> "event"; "feedback" -> "notify"; "framework" -> "framework"; "integration" -> "event"; "main" -> "framework"; "main" -> "task"; "notify" -> "common"; "task" -> "common"; "test_helpers" -> "common";
ステップ4: セルフループの削除 "event" -> "framework"; "feedback" -> "common"; "feedback" ->
"framework"; "feedback" -> "event"; "feedback" -> "notify"; "integration" -> "event"; "main" -> "framework"; "main" -> "task"; "notify" -> "common"; "task" -> "common"; "test_helpers" -> "common";
ステップ5: ビジュアライズ(初期状態準備完了)
② 探索
見たくないものを除去
見たくないものを除去
見たくないものを除去
詳しく見たいものを掘り下げる
詳しく見たいものを掘り下げる
注意
前提として 人に優しい粒度で パッケージ(階層)化 されてないと厳しい
デモ
背景 見えてくるもの 構造 と その重要性 可視化の 必要性
特徴的なパターン 依存しかしない: エントリーポイント/ルーター
特徴的なパターン 依存しかしない: エントリーポイント/ルーター 他のモジュールの 呼び分けを行う。 機能的な依存をしない ようにすると、変化の 影響を直接受けない ので怖くない。
特徴的なパターン 依存されるが他への依存が少ない(しない): アプリケーション基盤、共通部品
特徴的なパターン 依存されるが他への依存が少ない(しない): アプリケーション基盤、共通部品 変化させにくい部分。 必要とされる概念を きちんとモデリング、 一度作れば変化しなく て済むようにする。
特徴的なパターン 依存されるし、他への依存も多い: ビジネスロジック/コアドメイン
特徴的なパターン 依存されるし、他への依存も多い: ビジネスロジック/コアドメイン アプリケーションの 中核となる部分。 環境やビジネスの変 化に応じて変化して いく。
変 更 頻 度 依存されている数 多 い 少 な い
多い 少ない
変 更 頻 度 依存されている数 多 い 少 な い
多い 少ない アプリケーション基盤 エントリーポ イント ビジネス ロジック
変 更 頻 度 依存されている数 多 い 少 な い
多い 少ない アプリケーション基盤 ビジネス ロジック エントリーポ イント ここを 他と隔離しつつ きちんと保つ ことが重要
☠不吉な匂い
相互依存 / 循環参照 A B
相互依存 / 循環参照 A B
相互依存 / 循環参照 A B
相互依存 / 循環参照 A B
相互依存 / 循環参照 A B
相互依存 / 循環参照 A B
相互依存 / 循環参照 A B
相互依存 / 循環参照 A B
相互依存 / 循環参照
相互依存 / 循環参照
相互依存 / 循環参照
相互依存 / 循環参照
相互依存 / 循環参照
たいていは 責務違反
名前がない 隠れた役割を持つ
まとめ
まとめ 1. 構造とは一つの全体を構成する部分の関係・役割 2. 構造のあり方がソフトウェアの柔軟性に影響を及ぼす (リファクタリングはよい構造を作るための活動) 3. うまく可視化することで構造を捉える 4. 構造の特徴・パターンから中核となる場所を見つけ、そこを
きちんと保つことが重要