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

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

ただいまの
回答率

90.76%

  • PHP

    19247questions

    PHPは、Webサイト構築に特化して開発されたプログラミング言語です。大きな特徴のひとつは、HTMLに直接プログラムを埋め込むことができるという点です。PHPを用いることで、HTMLを動的コンテンツとして出力できます。HTMLがそのままブラウザに表示されるのに対し、PHPプログラムはサーバ側で実行された結果がブラウザに表示されるため、PHPスクリプトは「サーバサイドスクリプト」と呼ばれています。

  • Laravel 5

    1705questions

    Laravel 5は、PHPフレームワークLaravelの最新バージョンで、2014年11月に発表予定です。ディレクトリ構造がが現行版より大幅に変更されるほか、メソッドインジェクションやFormRequestの利用が可能になります。

  • Eloquent

    47questions

    Eloquentとは、PHPフレームワークのLaravelに最初から含まれているORM(Object-relational mapping:オブジェクト関係マッピング)です。

Eloquentのmorphとwithの併用は可能か?

解決済

回答 2

投稿 編集

  • 評価
  • クリップ 0
  • VIEW 266

masaya_ohashi

score 8722

EloquentのPolymorphic RelationsでHogeとFugaを使い分けているデータがあるとします。

 hoge

id name number
1 hoge 0
<?php
namespace App;
use Illuminate\Database\Eloquent\Model;
class Hoge extends Model
{
    public function base() {
        return $this->morphMany(Base::class, 'ref');
    }
}

 fuga

id name text
1 fuga a
<?php
namespace App;
use Illuminate\Database\Eloquent\Model;
class fuga extends Model
{
    public function base() {
        return $this->morphMany(Base::class, 'ref');
    }
}

 base

id ref_id ref_type
1 1 Hoge
2 1 Fuga
<?php
namespace App;
use Illuminate\Database\Eloquent\Model;
class Base extends Model
{
    public function ref() {
        return $this->morphTo();
    }
}

このとき、withメソッドを使わずBaseを得たときの挙動はこうなります。

$base_list = Base::all();
// [
//   ['id' => 1, 'ref_id' => 1, 'ref_type' => 'Hoge'],
//   ['id' => 2, 'ref_id' => 1, 'ref_type' => 'Fuga']
// ]

さらに、withを使うとこうなります。

$base_list = Base::with(['ref'])->all();
// [
//   ['id' => 1, 'ref_id' => 1, 'ref_type' => 'Hoge', 'ref' => [id  => 1, 'name' => 'hoge', 'number' => 0]],
//   ['id' => 2, 'ref_id' => 1, 'ref_type' => 'Fuga', 'ref' => [id  => 1, 'name' => 'fuga', 'text' => 'a']],
// ]

ここまではいいのですが、例えばHogeにだけ新たにImageモデルのリレーションを加えたとします。

 image

id hoge_id path
1 1 image.jpg
<?php
namespace App;
use Illuminate\Database\Eloquent\Model;
class Image extends Model
{
    public function hoge() {
        return $this->belongsTo();
    }
}

 Hoge(改変)

<?php
namespace App;
use Illuminate\Database\Eloquent\Model;
class Hoge extends Model
{
    public function base() {
        return $this->morphMany(Base::class, 'ref');
    }
    public function images() {
        return $this->hasMany(Image::class);
    }
}

このあと、HogeのImageも欲しいので以下のようにwithを書き直すとエラーになります。

$base_list = Base::with(['ref', 'ref.images'])->all();
// Hogeにはimagesがあるが、Fugaにはimagesのリレーションがないのでエラー

当たり前だ、と言われればそれまでなのですが、「あったら取る」「なければnull、またはプロパティ自体が無い」状態のデータが取得したいのです。なんとか方法はないでしょうか。

 試したこと

  • Hogeモデルのappendsにimageを加え、getImageAttributeを実装する
class Hoge extends Model
{
    public function base() {
        return $this->morphMany(Base::class, 'ref');
    }
    public function images() {
        return $this->hasMany(Image::class);
    }
    protected $appends = ['images'];
    public function getImagesAttribute($value) {
        return $this->images; // なぜかここでimagesというattributeは無いとエラーになる。
        // なお外部から$hoge->imagesとアクセスした場合はエラーにならない
    }
}
  • 気になる質問をクリップする

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

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

    クリップを取り消します

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

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

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

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

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

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

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

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

    質問の評価を下げる

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

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

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

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

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

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

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

    詳細な説明はこちら

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

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

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

回答 2

check解決した方法

0

試した方法の「$appendsにimageを加える」で、なぜエラーになるのかをコードを追っていったところ、以下のコードを見つけました。

// Illuminate\Database\Eloquent\Concerns\HasAttribute

    public function getAttribute($key)
    {
        if (! $key) {
            return;
        }

        // If the attribute exists in the attribute array or has a "get" mutator we will
        // get the attribute's value. Otherwise, we will proceed as if the developers
        // are asking for a relationship's value. This covers both types of values.
        if (array_key_exists($key, $this->attributes) ||
            $this->hasGetMutator($key)) {
            return $this->getAttributeValue($key);
        }

        // Here we will determine if the model base class itself contains this given key
        // since we do not want to treat any of those methods are relationships since
        // they are all intended as helper methods and none of these are relations.
        if (method_exists(self::class, $key)) {
            return;
        }

        return $this->getRelationValue($key);
    }


このgetAttributeというメソッドはモデルに対し__getの呼び出し時に呼ばれるメソッドで、$appendsで加えたいプロパティ名と、リレーションのメソッド名が同一である場合、以下のifに引っかかってしまうようです。

        if (method_exists(self::class, $key)) {
            return;
        }

なので、このgetAttributeの最後で呼ばれているgetRelationValueを直接呼べばいいんじゃないか?と思って試したところ、希望通りの動作をしました。

class Hoge extends Model
{
    public function base() {
        return $this->morphMany(Base::class, 'ref');
    }
    public function images() {
        return $this->hasMany(Image::class);
    }
    protected $appends = ['images'];
    public function getImagesAttribute($value) {
        return $this->getRelationValue('images');
    }
}


やはり多少面倒でもコードを追うことは大事だということをまた実感させられました。

投稿

編集

  • 回答の評価を上げる

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

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

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

  • 回答の評価を下げる

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

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

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

  • 2017/10/04 11:15

    嫌がらせでマイナスつけていくのやめてもらえませんかね…

    キャンセル

-1

以下のように、一度対象のプロパティにアクセスすれば、強引に値を読み込ませることができ、以降はimageもtoArrayに含まれるようになりました。

$base_list = Base::with(['ref'])->all();
foreach($base_list as $index => $base) {
    if($base->ref_type == 'Hoge') {
        $base->images;
    }
}
// [
//   ['id' => 1, 'ref_id' => 1, 'ref_type' => 'Hoge', 'ref' => [id  => 1, 'name' => 'hoge', 'number' => 0, 'images' => [['id' => 1, 'hoge_id' => 1, 'path' => 'image.jpg']]]],
//   ['id' => 2, 'ref_id' => 1, 'ref_type' => 'Fuga', 'ref' => [id  => 1, 'name' => 'fuga', 'text' => 'a']],
// ]

とても泥臭いので、もう少しだけ他の方の意見を待ちたいと思います。

投稿

編集

  • 回答の評価を上げる

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

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

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

  • 回答の評価を下げる

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

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

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

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

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

関連した質問

  • 解決済

    PHPのオペコードについて教えてください

    現在、関数の勉強をしておりまして、理解を深めるためにオペコードレベルでどのように処理されているのかを調べております。 「PHPはどのように動くのか ~PHPコアから読み解く仕組み

  • 解決済

    CakePHP2:テーブルモデルの共有は可能?

    接頭辞のみが異なる、まったく同じテーブル構成のテーブル[接頭辞]_applesがあります。 これらのテーブルで同じテーブルモデルの共有は可能でしょうか? 現在、以下のように

  • 解決済

    Laravel5- 動的スコープの呼び出しについて

    お世話になります。 今回、Eloquentを利用して、データベースからselectしてきた値をviewに表示させるプログラムを書きたいと思っています。 入力されたデータをス

  • 解決済

    sql文 カラムを一緒にしたい。

    このようなテーブルがあった場合 FOLLOW_IDをUSER_IDとして認識させることはできますか?? つまりFOLLOW_IDの値をUSER_IDのカラムに足した

  • 解決済

    php 継承関係について

    お世話になっております。 PHPの継承の関係について質問させてください。 ある理由がありまして、継承先から継承元にメソッドを渡したいのですがどうしたら良いでしょうか。 元のファ

  • 解決済

    PHPのforeach文について

    前提・実現したいこと echo $course_name_array[2]; の下のforeach文を正常に動作させたいです 発生している問題・エラーメッセージ Notice:

  • 解決済

    【Laravel】ソフトデリートカラムのカスタマイズ

    テーブルにdeleted_atカラムを追加するソフトデリート機能を有効にできると思いますが、 このカラム名をdelete_flag(1=削除)のように独自のカラムに変更することは可

  • 解決済

    MySQLのカラムに収めるデータ形式について

    お世話になります。 勉強しながらPHPとMySQLの連携で小規模なウェブサイトを作ったりしている身ですが、新しくアプリを設計するとき、表題にある「カラムに収めるデータの形式」につい

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

  • PHP

    19247questions

    PHPは、Webサイト構築に特化して開発されたプログラミング言語です。大きな特徴のひとつは、HTMLに直接プログラムを埋め込むことができるという点です。PHPを用いることで、HTMLを動的コンテンツとして出力できます。HTMLがそのままブラウザに表示されるのに対し、PHPプログラムはサーバ側で実行された結果がブラウザに表示されるため、PHPスクリプトは「サーバサイドスクリプト」と呼ばれています。

  • Laravel 5

    1705questions

    Laravel 5は、PHPフレームワークLaravelの最新バージョンで、2014年11月に発表予定です。ディレクトリ構造がが現行版より大幅に変更されるほか、メソッドインジェクションやFormRequestの利用が可能になります。

  • Eloquent

    47questions

    Eloquentとは、PHPフレームワークのLaravelに最初から含まれているORM(Object-relational mapping:オブジェクト関係マッピング)です。