質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

ただいまの
回答率

88.07%

【Laravel】Eloquentモデルのマージについて

解決済

回答 1

投稿

  • 評価
  • クリップ 1
  • VIEW 9,163

score 38

Laravel5.5を使用しています。

LaravelのEloquentを用いて、取得したコレクションをマージした際、想定した動作になりません。
以下、サンプルコードです。

$hoge = new Hoge(); // Eloquentモデル

$result1 = $hoge->select('hoge.*')->join('huga','hoge.id', '=', 'huga.hoge_id')->get();
/**
検索結果は3件で以下のような形
object(Illuminate\Database\Eloquent\Collection)[80]
  protected 'items' => 
    array (size=3)
      0 => object(App\Models\Hoge), // id:1
      1 => object(App\Models\Hoge), // id:2
      2 => object(App\Models\Hoge)  // id:3
*/


$result2 = $hoge->select('hoge.*')->join('piyo','hoge.id', '=', 'piyo.hoge_id')->get();
/**
検索結果は2件で以下のような形
object(Illuminate\Database\Eloquent\Collection)[80]
  protected 'items' => 
    array (size=2)
      0 => object(App\Models\Hoge), // id:4
      1 => object(App\Models\Hoge)  // id:5
*/


$result1->merge($result2);
/**
求める形は以下
object(Illuminate\Database\Eloquent\Collection)[80]
  protected 'items' => 
    array (size=5)
      0 => object(App\Models\Hoge), // id:1
      1 => object(App\Models\Hoge), // id:2
      2 => object(App\Models\Hoge)  // id:3
      3 => object(App\Models\Hoge), // id:4
      4 => object(App\Models\Hoge)  // id:5
*/

しかし実際に出力された値は

$result1->merge($result2);
/**
求める形は以下
object(Illuminate\Database\Eloquent\Collection)[80]
  protected 'items' => 
    array (size=1)
      4 => object(App\Models\Hoge)  // id:5
*/

  

以下のようにコードを変更すると求める値が取得できました。

collect($result1->all())->merge($result2);

  

しかしマージのために新たに(しかも一方だけ)コレクションクラスでラップするのも見栄えが悪く無駄な処理をしている気がします。

何か
$result1->xxx($result2);
のように簡潔な形で求めるコレクションを取得する方法はないでしょうか。

よろしくお願いいたします。

  • 気になる質問をクリップする

    クリップした質問は、後からいつでもマイページで確認できます。

    またクリップした質問に回答があった際、通知やメールを受け取ることができます。

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 過去に投稿した質問と同じ内容の質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

回答 1

checkベストアンサー

+2

Laravel5.5で、コレクションのconcatメソッドを利用すると2つが結合したデータを得られるかと思います。
$result1->concat($result2);
Laravel 5.5 コレクション concat()

Illuminate/SupportのCollectionクラスと、それを継承するlluminate/Database/EloquentのCollectionでは、mergeメソッドの実装が異なっているみたいなので注意が必要ですね。
Illuminate/Support\Collection
lluminate/Database/Eloquent

    //以下は添字配列の場合は、結果が違う
    $items = $data1->merge($data2); //キー重複分が上書き
    $items = collect($data1)->merge($data2); //array_mergeを内部で利用

    dd($items->toArray());

投稿

編集

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2017/10/24 20:41

    期待した結果を得ることができました。ありがとうございます。

    キャンセル

15分調べてもわからないことは、teratailで質問しよう!

  • ただいまの回答率 88.07%
  • 質問をまとめることで、思考を整理して素早く解決
  • テンプレート機能で、簡単に質問をまとめられる

関連した質問

同じタグがついた質問を見る