Documentation Index
Fetch the complete documentation index at: https://kawax.biz/llms.txt
Use this file to discover all available pages before exploring further.
コレクションとは
Illuminate\Support\Collection クラスは、配列データを操作するための流暢なラッパーです。
PHPの標準的な配列関数を個別に呼び出す代わりに、メソッドチェーンで直感的にデータを加工できます。
// 通常の配列操作
$names = array_filter(
array_map(fn ($user) => $user['name'], $users),
fn ($name) => $name !== null
);
// コレクションを使うと
$names = collect($users)
->pluck('name')
->filter()
->values();
コレクションはイミュータブル(不変)です。各メソッドは元のコレクションを変更せず、新しいコレクションインスタンスを返します。元のデータを保持しながら安全に操作できます。
コレクションの作成
collect() ヘルパー
最もよく使われる方法です。配列を渡してコレクションを生成します。
use Illuminate\Support\Collection;
// 配列からコレクションを作成
$users = collect([
['name' => '山田太郎', 'age' => 28, 'role' => 'admin'],
['name' => '鈴木花子', 'age' => 34, 'role' => 'editor'],
['name' => '佐藤次郎', 'age' => 22, 'role' => 'viewer'],
]);
// ネストした配列もOK
$products = collect([
['name' => 'ノートPC', 'price' => 120000, 'stock' => 5],
['name' => 'マウス', 'price' => 3500, 'stock' => 20],
['name' => 'キーボード', 'price' => 8000, 'stock' => 12],
]);
Collection::make()
collect() と同等です。ファサードスタイルで書きたい場合に使います。
$collection = Collection::make([1, 2, 3]);
Collection::fromJson()
JSON文字列からコレクションを作成します。外部APIのレスポンスを処理する場合に便利です。
$collection = Collection::fromJson('[{"name":"山田太郎"},{"name":"鈴木花子"}]');
よく使うメソッド
map — 各要素を変換する
コレクションの各要素に処理を適用し、変換された新しいコレクションを返します。
$users = collect([
['name' => '山田太郎', 'email' => '[email protected]'],
['name' => '鈴木花子', 'email' => '[email protected]'],
]);
// メールアドレスをマスクして表示用に変換
$masked = $users->map(function (array $user) {
$parts = explode('@', $user['email']);
return [
'name' => $user['name'],
'email' => substr($parts[0], 0, 2) . '***@' . $parts[1],
];
});
// [['name' => '山田太郎', 'email' => 'ya***@example.com'], ...]
filter / reject — 条件で絞り込む
filter() は条件を満たす要素だけを残し、reject() は条件を満たす要素を除外します。
$products = collect([
['name' => 'ノートPC', 'price' => 120000, 'in_stock' => true],
['name' => 'マウス', 'price' => 3500, 'in_stock' => false],
['name' => 'キーボード', 'price' => 8000, 'in_stock' => true],
]);
// 在庫ありの商品だけ取得
$inStock = $products->filter(fn ($product) => $product['in_stock']);
// 在庫なしの商品を除外(reject は filter の反転)
$available = $products->reject(fn ($product) => ! $product['in_stock']);
// 引数なしの filter() は falsy な値を除去
$names = collect(['山田', '', null, '鈴木', false])->filter()->values();
// ['山田', '鈴木']
filter() 後はインデックスが歯抜けになります。values() を呼ぶと0始まりの連番に振り直されます。
first / last — 要素を1件取得する
条件を満たす最初または最後の要素を返します。
$orders = collect([
['id' => 1, 'status' => 'shipped', 'amount' => 5000],
['id' => 2, 'status' => 'pending', 'amount' => 12000],
['id' => 3, 'status' => 'pending', 'amount' => 3500],
]);
// 最初の未処理注文を取得
$nextOrder = $orders->first(fn ($order) => $order['status'] === 'pending');
// ['id' => 2, 'status' => 'pending', 'amount' => 12000]
// 条件に一致しない場合のデフォルト値
$order = $orders->first(fn ($order) => $order['status'] === 'cancelled', null);
// 最後の要素
$latest = $orders->last();
pluck — 特定のキーの値だけ取り出す
ネストした配列やモデルから特定のフィールドだけを抜き出します。
$users = collect([
['id' => 1, 'name' => '山田太郎', 'department' => '開発部'],
['id' => 2, 'name' => '鈴木花子', 'department' => '営業部'],
['id' => 3, 'name' => '佐藤次郎', 'department' => '開発部'],
]);
// 名前だけ取り出す
$names = $users->pluck('name');
// ['山田太郎', '鈴木花子', '佐藤次郎']
// id をキーにしてマッピング
$nameById = $users->pluck('name', 'id');
// [1 => '山田太郎', 2 => '鈴木花子', 3 => '佐藤次郎']
groupBy — 特定のキーでグループ化する
$users = collect([
['name' => '山田太郎', 'department' => '開発部'],
['name' => '鈴木花子', 'department' => '営業部'],
['name' => '佐藤次郎', 'department' => '開発部'],
['name' => '田中美咲', 'department' => '営業部'],
]);
$byDepartment = $users->groupBy('department');
// [
// '開発部' => [['name' => '山田太郎', ...], ['name' => '佐藤次郎', ...]],
// '営業部' => [['name' => '鈴木花子', ...], ['name' => '田中美咲', ...]],
// ]
// クロージャでグループキーを動的に指定
$byFirstChar = $users->groupBy(fn ($user) => mb_substr($user['name'], 0, 1));
sortBy / sortByDesc — 並び替える
$products = collect([
['name' => 'ノートPC', 'price' => 120000],
['name' => 'マウス', 'price' => 3500],
['name' => 'キーボード', 'price' => 8000],
]);
// 価格の昇順
$cheapFirst = $products->sortBy('price');
// 価格の降順
$expensiveFirst = $products->sortByDesc('price');
// 複数キーで並び替え
$sorted = $products->sortBy([
['price', 'asc'],
['name', 'asc'],
]);
each — 各要素に処理を実行する
副作用のある処理(ログ出力、メール送信など)に使います。コレクション自体は変更しません。
$orders = collect([
['id' => 1, 'user_id' => 10, 'amount' => 5000],
['id' => 2, 'user_id' => 11, 'amount' => 12000],
]);
$orders->each(function (array $order) {
\Log::info("注文 #{$order['id']} を処理中", ['amount' => $order['amount']]);
// 通知送信やキューへのディスパッチなど
});
// false を返すと途中で処理を抜けられる
$orders->each(function (array $order) {
if ($order['amount'] > 10000) {
return false; // ループを抜ける
}
// ...
});
flatMap — マップ後に1段階フラット化する
各要素を配列に変換し、全体を1次元に平坦化します。
$users = collect([
['name' => '山田太郎', 'tags' => ['php', 'laravel']],
['name' => '鈴木花子', 'tags' => ['javascript', 'vue']],
]);
// 全ユーザーのタグをフラットなリストに
$allTags = $users->flatMap(fn ($user) => $user['tags']);
// ['php', 'laravel', 'javascript', 'vue']
reduce — 集計する
コレクション全体を1つの値に畳み込みます。
$orders = collect([
['product' => 'ノートPC', 'quantity' => 1, 'price' => 120000],
['product' => 'マウス', 'quantity' => 2, 'price' => 3500],
['product' => 'キーボード', 'quantity' => 1, 'price' => 8000],
]);
// 合計金額を計算
$total = $orders->reduce(
fn ($carry, $order) => $carry + ($order['price'] * $order['quantity']),
0
);
// 135000
単純な合計なら sum() が使えます: $orders->sum(fn ($o) => $o['price'] * $o['quantity'])
chunk — 分割する
大きなコレクションを指定サイズに分割します。バッチ処理や画面表示で便利です。
$users = collect(range(1, 100))->map(fn ($i) => ['id' => $i, 'name' => "ユーザー{$i}"]);
// 10件ずつに分割
$chunks = $users->chunk(10);
// $chunks->count() === 10
// バッチごとに処理
$chunks->each(function (\Illuminate\Support\Collection $batch) {
// バッチごとのDB操作やメール送信など
});
メソッドチェーン
コレクションの真価はメソッドチェーンにあります。複数の操作をつなげて、複雑なデータ変換を1つの式で表現できます。
$orders = collect([
['customer' => '山田太郎', 'status' => 'completed', 'amount' => 5000, 'category' => '電子機器'],
['customer' => '鈴木花子', 'status' => 'pending', 'amount' => 12000, 'category' => '電子機器'],
['customer' => '佐藤次郎', 'status' => 'completed', 'amount' => 3500, 'category' => '文具'],
['customer' => '田中美咲', 'status' => 'completed', 'amount' => 8000, 'category' => '電子機器'],
['customer' => '伊藤健一', 'status' => 'cancelled', 'amount' => 2000, 'category' => '文具'],
]);
// 完了済みの電子機器注文を金額降順で取得し、顧客名と金額だけ抽出する
$result = $orders
->filter(fn ($order) => $order['status'] === 'completed')
->filter(fn ($order) => $order['category'] === '電子機器')
->sortByDesc('amount')
->map(fn ($order) => [
'customer' => $order['customer'],
'amount' => number_format($order['amount']) . '円',
])
->values();
// [
// ['customer' => '田中美咲', 'amount' => '8,000円'],
// ['customer' => '山田太郎', 'amount' => '5,000円'],
// ]
Eloquentとの連携
Eloquentクエリの結果は常に Illuminate\Database\Eloquent\Collection インスタンスとして返されます。
これは基本の Collection を継承しており、上記のメソッドがすべて使えます。
use App\Models\User;
use App\Models\Order;
// get() の結果はコレクション
$users = User::where('is_active', true)->get(); // Collection
// コレクションのメソッドをそのまま使える
$adminEmails = User::all()
->filter(fn ($user) => $user->role === 'admin')
->pluck('email');
// リレーションのロード済みデータもコレクション操作できる
$orders = Order::with('items')->where('status', 'completed')->get();
$summary = $orders->map(fn ($order) => [
'id' => $order->id,
'customer' => $order->user->name,
'item_count' => $order->items->count(),
'total' => $order->items->sum('price'),
]);
データベースで処理できるフィルタリングや並び替えは、コレクション操作ではなくクエリビルダ(where、orderBy など)で行いましょう。コレクションへの変換後にフィルタリングすると、不要なデータをすべてメモリに読み込んでしまいます。
Eloquent固有のコレクションメソッド
Eloquent\Collection には追加のメソッドがあります。
$users = User::all();
// モデルをIDで検索
$user = $users->find(1);
// 主キーのコレクションを取得
$ids = $users->modelKeys(); // [1, 2, 3, ...]
// リレーションをまとめてロード
$users->load('orders', 'profile');
// 差集合・積集合
$diff = $users->diff($otherUsers);
$intersect = $users->intersect($otherUsers);
Lazy Collections
通常のコレクションはデータ全体をメモリに読み込みますが、LazyCollection はPHPのジェネレーターを活用し、データを1件ずつ処理します。
数万件以上の大量データを扱う場合に、メモリを節約できます。
use Illuminate\Support\LazyCollection;
// 通常のコレクション: 全データをメモリに読み込む
$users = User::all(); // 10万件あれば10万件がメモリに乗る
// LazyCollection: 1件ずつ処理
User::cursor()->each(function (User $user) {
// ユーザーを1件ずつ処理
// メモリに保持するのは常に1件分だけ
});
LazyCollectionの作成
use Illuminate\Support\LazyCollection;
// クロージャ(ジェネレーター)から作成
$lazy = LazyCollection::make(function () {
$handle = fopen('large-file.csv', 'r');
while (($line = fgetcsv($handle)) !== false) {
yield $line;
}
fclose($handle);
});
// Eloquentの cursor() メソッドが LazyCollection を返す
$lazy = User::where('is_active', true)->cursor();
大量データのバッチ処理
use App\Models\Order;
// 全注文を1件ずつ処理(メモリ効率が良い)
Order::cursor()
->filter(fn ($order) => $order->amount > 10000)
->each(fn ($order) => $order->sendConfirmationEmail());
// cursor() で全件を1件ずつフェッチしつつ filter/each でパイプライン処理
Order::where('status', 'completed')
->cursor()
->filter(fn ($order) => $order->amount > 10000)
->each(fn ($order) => $order->sendConfirmationEmail());
cursor() はデータベースから1件ずつフェッチするため、大量のレコードを処理する際にメモリを大幅に節約できます。ただし、データベース接続は処理の完了まで維持されます。
take と skip でページング
$lazy = LazyCollection::make(function () {
foreach (range(1, 1000000) as $i) {
yield $i;
}
});
// 最初の1000件をスキップして、次の100件を取得
$page = $lazy->skip(1000)->take(100)->values();
まとめ
| メソッド | 説明 |
|---|
collect($array) | コレクションを作成する |
map($callback) | 各要素を変換する |
filter($callback) | 条件で絞り込む |
reject($callback) | 条件に合う要素を除外する |
first($callback) | 条件を満たす最初の要素を取得 |
pluck($key) | 特定キーの値を抜き出す |
groupBy($key) | 特定キーでグループ化 |
sortBy($key) | 昇順で並び替え |
sortByDesc($key) | 降順で並び替え |
each($callback) | 各要素に処理を実行(副作用) |
flatMap($callback) | マップ後にフラット化 |
reduce($callback, $initial) | 集計して1つの値にする |
chunk($size) | 指定サイズに分割 |
values() | インデックスを振り直す |
sum($key) | 合計値を計算 |
count() | 件数を取得 |
コレクションは配列関数よりもはるかに可読性が高くなります。// 配列関数を使った場合
$result = array_values(array_filter(
array_map(fn ($u) => $u['name'], $users),
fn ($name) => strlen($name) > 2
));
// コレクションを使った場合
$result = collect($users)
->pluck('name')
->filter(fn ($name) => strlen($name) > 2)
->values()
->all();
通常のコレクション vs LazyCollection
- 通常のコレクション: 数百〜数千件程度のデータ。シンプルで直感的。
- LazyCollection: 数万件以上の大量データ。メモリを節約したい場合。
cursor() と組み合わせると効果的。
- Eloquentの
get() は通常のコレクション、cursor() は LazyCollection を返します。