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
Qiita Night PHP 2023
Search
Sponsored
·
Your Podcast. Everywhere. Effortlessly.
Share. Educate. Inspire. Entertain. You do you. We'll handle the rest.
→
ふわせぐ
January 26, 2023
Programming
15k
0
Share
Embed
Copy iframe code
Copy JS code
Copy link
Start on current slide
Qiita Night PHP 2023
登壇資料です
ふわせぐ
January 26, 2023
More Decks by ふわせぐ
See All by ふわせぐ
php-conference-nagoya-2025
fuwasegu
0
530
object-oriented-conference-2024
fuwasegu
14
5.7k
proud of my organization LT
fuwasegu
1
470
php conference okinawa 2022
fuwasegu
0
2k
QiitaConference2022
fuwasegu
1
1.5k
sleepagotchi
fuwasegu
2
2.7k
何の画像か当てちゃるBot紹介/image_guess_bot
fuwasegu
0
230
新卒 Laravel 初心者が成長していく中で 感じたコレジャナイ感/PHPerKaigi 2022
fuwasegu
11
16k
入社初日に社内サービスを全部一人で引き継いだ新卒フルサイクルエンジニアの話
fuwasegu
4
7k
Other Decks in Programming
See All in Programming
AIとASP.NET Coreで雑Webアプリを作った話
mayuki
0
520
並列実装の現場、2ヶ月間実務でAIを使い倒したAIもPCも私も限界が近い
ming_ayami
0
130
生成AI時代にこそ効くGo | Why Go Works in the Age of Generative AI
mom0tomo
8
3.2k
ADKを使って簡単にAIエージェントを作ってみよう
k1mu21
0
260
Agentic UI
manfredsteyer
PRO
0
150
依存関係から依存物へ―Dependencyという言葉の歴史をひも解く
j_lee
0
120
Technical Debt: Understanding it Rightly, Engaging it Rightly #LaravelLiveJP
shogogg
0
220
3Dシーンの圧縮
fadis
1
760
フロントエンドとバックエンドで「1文字」を揃えよう
youkidearitai
PRO
0
500
作って学ぶ、 JSX (TSX) ランタイムの基本
syumai
7
1.6k
Creating Composable Callables in Contemporary C++
rollbear
0
110
Go1.27で導入されるジェネリクスメソッドでできること
mackee
0
110
Featured
See All Featured
Marketing Yourself as an Engineer | Alaka | Gurzu
gurzu
0
230
Bioeconomy Workshop: Dr. Julius Ecuru, Opportunities for a Bioeconomy in West Africa
akademiya2063
PRO
1
140
Why Your Marketing Sucks and What You Can Do About It - Sophie Logan
marketingsoph
0
170
How People are Using Generative and Agentic AI to Supercharge Their Products, Projects, Services and Value Streams Today
helenjbeal
1
210
世界の人気アプリ100個を分析して見えたペイウォール設計の心得
akihiro_kokubo
PRO
71
40k
Code Reviewing Like a Champion
maltzj
528
40k
svc-hook: hooking system calls on ARM64 by binary rewriting
retrage
2
300
The Spectacular Lies of Maps
axbom
PRO
1
800
A better future with KSS
kneath
240
18k
[SF Ruby Conf 2025] Rails X
palkan
2
1.1k
The untapped power of vector embeddings
frankvandijk
2
1.8k
Understanding Cognitive Biases in Performance Measurement
bluesmoon
32
2.9k
Transcript
෮ॲཧͱ JUFSBCMFͰ༇շͳؒͨͪ Hirosugu Takeshita @fuwasegu
• 竹下 拓秀 / ふわせぐ (@fuwasegu) • 株式会社ゆめみ / 21卒
– コーポレートエンジニア • PHP / Laravel がメイン – Svelte(JS / TS),Rust Go 勉強中 • 長崎県出身 / 愛知県在住 • !(3 歳)の父 2 自己紹介 Qiita Night 〜PHP〜
宣伝 3 Qiita Night 〜PHP〜 • mpyw さんと二人で Laravel の
ソースコードを読む会 • 隔週で YouTube Live で配信中 • バックナンバー • Model の with • Model の scope • SoftDelete • サービスコンテナ • ルートモデルバインディング • ExceptionHandler 次回: 2023-01-30 14:00:00
• PHP での反復処理について • Generator の紹介 4 今日話すこと Qiita Night
〜PHP〜
5 7 分しかないのでめっちゃ突っ走ります (後日 Qiita に記事出すので詳しくはそちらで) Qiita Night 〜PHP〜
反復処理(Iteration)ってどんなもの? Qiita Night 〜PHP〜 6 反復可能な値の各要素に対して,同じ処理を順番に行うこと
• オブジェクト • iterable 型 7 PHP で反復可能な値になりうるもの Qiita Night
〜PHP〜
• オブジェクト 8 PHP で反復可能な値になりうるもの Qiita Night 〜PHP〜 アクセス権限があるプロパティは, foreach
で取り出せる (private なものは取り出せない)
• iterable 型 9 PHP で反復可能な値になりうるもの Qiita Night 〜PHP〜 今日の本題はこっち
• iterable = (Traversable | array) – 型のエイリアス – PHP7.1
〜 PHP8.0 では疑似型 – それ以前は未定義 • foreach で反復処理ができるもの • iterator_to_array() に渡すと array になるもの – iterator_to_array() の引数 Traversable | array になったのは PHP8.2 から – それまでは Traversable のみだった 10 iterable とは Qiita Night 〜PHP〜
11 Qiita Night 〜PHP〜 iterable array Traversable
12 Qiita Night 〜PHP〜 iterable array Traversable IteratorAggregate Iterator Generator
extends extends extends implements
13 Qiita Night 〜PHP〜 iterable IteratorAggregate Iterator Generator extends extends
extends implements 時間がないので説明は割愛
Iterator とは • 反復処理中の状態を持つオブジェクト自身を 定義するinterface – 現在の要素を取得 – 現在のキーを取得 –
次の要素に進む – 最初の要素に巻き戻し – 現在位置が有効かどうか 取得 Qiita Night 〜PHP〜 14
IteratorAggregate とは • Iterator(Traversable) を作る interface • 自分自身が要素を持ったり状態を管理するわけ ではない Qiita
Night 〜PHP〜 15 • getIterator() は新しいイテレータ を返す • foreach に IteratorAggregate を渡 すと勝手に getIterator() が呼ばれ る ポイント
Generator とは • イテレータが簡単に作れるやつ – 本来 Iterator を自前で作る場合,インタフェースに定 義されたメソッドをそれぞれ実装する必要がある –
Generator は yield するだけで良い • 遅延評価される(= 省メモリ) – 値(演算結果)が必要になったときに初めて評価 されるということ Qiita Night 〜PHP〜 16
17 ジェネレーター構文の例 Qiita Night 〜PHP〜
18 ジェネレーター構文の例 Qiita Night 〜PHP〜 まだ何が嬉しいか分からん!
19 もうちょっと 具体的なユースケースを考えます Qiita Night 〜PHP〜
仮想の要件 • 大容量の CSV ファイルを読み込む – ID,姓,名,生年月日,郵便番号,電話番号,メールアドレ ス,性別 • フィルターしてデータを抜き出す
– 名字が山田 – 二十歳以上 – 男 – 頭から 3 件取得する Qiita Night 〜PHP〜 20
愚直に実装してみる(方針) • 一気には読み込めないので一行ずつ読む – fopen() して fgetcsv() しながらループ • ループの中で
条件に合致しなかったら読み 飛ばす • カウンタを作っておいてインクリメントしていく – 必要数集まったら break Qiita Night 〜PHP〜 21
22 Qiita Night 〜PHP〜 条件の数だけ if を並べる 1個の if にまとめる
愚直に実装してみる(実装) 文字が小さいのは許して
23 Qiita Night 〜PHP〜 愚直に実装してみる(実装)
24 メソッドに分割してみる Qiita Night 〜PHP〜 年齢でフィルター 名前でフィルター 性別でフィルター 指定要素数取り出し
• 一見見通しが良くなったが,前もって CSV を全部読み込む必要があり激重 • 毎回 foreach で全探索するので激重 25 分割したメソッドで組み立て
Qiita Night 〜PHP〜 CSV が大きいとメモリが食いつぶされます
26 Generator を使って書き直す Qiita Night 〜PHP〜 年齢でフィルター 性別でフィルター 名前でフィルター 指定要素数取り出し
CSV 読み込み
27 Generator を使って書き直す Qiita Night 〜PHP〜 今回は Generator 関数を用いましたが,IteratorAggregate を
使ってクラスごとに分割する方法もあります (Qiita で解説します!)
• さっきと見た目は変わらない • 全て Generator なので CSV を一気に読み 込む必要も無くなった 28
分割したメソッドで組み立て Qiita Night 〜PHP〜 CSV がどれだけ大きくても動作します
• Generator を使わない場合 – Fatal error: Allowed memory size of
134217728 bytes exhausted (tried to allocate 20480 bytes) • Generator を使った場合 – 2936 バイトで正常に終了 29 実験 Qiita Night 〜PHP〜 • 100 万行の CSV(77.2 MB)を入力 • CSV の読み込み 〜 フィルターにかかる使用メモリを計測 結果
• Generator は yield が呼ばれるたびに値を生成する – Generator でない場合要素分のメモリが必要 – Generator
が生成した値は消費されるので古いものは破棄される – つまり巻き戻せない(単一方向の Iterator) • Generator は値が必要になったときに初めて評価される – 並列で並べた時,Generator でない場合毎回全部ループして都度 全要素評価する – Generator の場合,一行読み込むたびに次の処理に移れる 30 なぜ Generator は省メモリなのか Qiita Night 〜PHP〜 メモリの使用量が要素数に比例しないので,無限長のリストを扱える
31 実際の処理を見てみよう Qiita Night 〜PHP〜 6, 2, 10, 9, 3,
12 読み取りたいデータ <?php $data = read_data(); // 5 以上の整数に絞る $result = filter_1($data); // 偶数に絞る $result = filter_2($result); // 頭から 2 つだけとる $result = take($result, 2); foreach ($result as $item) { echo $item; } 擬似コード
32 Generator を使わない場合 Qiita Night 〜PHP〜 全部読み込み 全探索で絞り込み 全探索で絞り込み 早期リターン
出力 <?php $data = read_data(); // 5 以上の整数に絞る $result = filter_1($data); // 偶数に絞る $result = filter_2($result); // 頭から 2 つだけとる $result = take($result, 2); foreach ($result as $item) { echo $item; }
33 Generator を使わない場合 Qiita Night 〜PHP〜 6 2 10 9
3 12 read_data 1 2 3 4 5 6 filter_1 7 8 9 10 11 12 filter_2 13 14 15 16 take 17 18 foreach 19 20 データ 処理 黒の数字は読み取り順 処理が軸になる
34 Generator を使う場合 Qiita Night 〜PHP〜 1 要素目 2 要素目
3 要素目 <?php $data = read_data(); // 5 以上の整数に絞る $result = filter_1($data); // 偶数に絞る $result = filter_2($result); // 頭から 2 つだけとる $result = take($result, 2); foreach ($result as $item) { echo $item; }
35 Generator を使う場合 Qiita Night 〜PHP〜 6 2 10 9
3 12 read_data 1 6 8 filter_1 2 7 9 filter_2 3 10 take 4 11 foreach 5 12 データ 処理 黒の数字は読み取り順 値が軸になる
• Generator を使わない場合 – 全てのデータを読み込んだ – メソッドは呼び出し時に実行された – 毎回全要素を参照した •
take の早期リターンは例外 • Generator を使った場合 – 全てのデータは読み込まなかった – メソッドは呼び出し時に実行されなかった • 要素ごとに遅延実行された – 演算対象の要素のみ参照した 36 結果 Qiita Night 〜PHP〜 <?php $data = read_data(); // 5 以上の整数に絞る $result = filter_1($data); // 偶数に絞る $result = filter_2($result); // 頭から 2 つだけとる $result = take($result, 2); foreach ($result as $item) { echo $item; }
37 Qiita Night 〜PHP〜 Generator はいいぞ!