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

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

ただいまの
回答率

88.80%

【cakePHP3】クエリビルダでSELECT * したい【JOIN】

受付中

回答 2

投稿 編集

  • 評価
  • クリップ 0
  • VIEW 4,343

HelloWorld2

score 24

 概要

cakePHP3のクエリビルダを用いて、SQL文を発行しようとしています。
発行したいSQLは下記の通りです。
課題は、joinしたテーブルのカラムが取得できないことです。
(下記コードの、test_2、test_3のカラムが取得できない)

 発行したいSQL

SELECT * 
FROM   test_1 
INNER JOIN test_2 
        ON test_1.id = test_2.parent_id 
INNER JOIN test_3 
        ON test_1.id = test_3.parent_id 
WHERE  test_1.id = $id/*パラメータ、今回は'5'とする*/ 
LIMIT  1 

 現状

上記SQLを発行するため、下記コードを作成しました。

$this->test_1->find()
             ->join([
                 'test_2' => [
                     'table' => 'test_2',
                     'type' => 'INNER',
                     'conditions' => 'test_2.parent_id = test_1.id',
                 ],
                 'test_3' => [
                     'table' => 'test_3',
                     'type' => 'INNER',
                     'conditions' => 'test_3.parent_id = test_1.id',
                 ]
             ])
             ->where([
                 'id' => $id
             ])
             ->first()
             ;

結果として下記SQLが発行されました。

SELECT
    test_1.id AS `test_1__id`,
    test_1.name AS `test_1__name`,
    ・
    ・
    ・
    /*test_1の全カラムが出力されております*/
    /*test_2、test_3のカラムは出力されておりません*/
FROM 
  test_1 test_1 
  INNER JOIN test_2 test_2 ON test_2.parent_id = test_1.id 
  INNER JOIN test_3 test_3 ON test_3.parent_id = test_1.id 
WHERE 
  id = 5 
LIMIT 1

test_2、test3のカラムを取得するにはどうすればよいのでしょうか。

find('all')にしても変化はありませんでした。
また、参考サイトのように作成すると、エラーが出力されました。(単にcakePHPのバージョンの問題だと思います・・・)

計数百カラムあるので、select()内で、1つ1つカラムを指定する方法はNGです。

よろしくお願いします。

 補足

 アソシエーション定義

定義を作成してみましたが、うまくいきません。

namespace App\Model\Table;

use Cake\ORM\Table;

class test_1Table extends Table
{
    public function initialize(array $config)
    {
        $this->hasOne('test_2')
            -> setJoinType('INNER')
            -> setForeignKey('parent_id')
            -> setBindingKey('id')
        ;
        /*配列で記載しても変わらず
        $this->hasOne('cmp_tran_purpose',[
            'joinType' => 'INNER',
            'foreignKey' => 'parent_id',
            'bindingKey' => 'id'
        ]);
        */
    }
}

エラーメッセージ

The test_2 association is not defined on test_1. 

 疑問点と試行錯誤

  1. test_2のテーブルクラスは作成する必要ありますか?(作成してみましたが、上記と同じエラーメッセージ)
  2. test_2にinitializeメソッドは必要ですか?(作成してみましたが、上記と同じエラーメッセージ)
  3. Entityクラスは作成する必要がありますか?また、クラス内で何か記述する必要はありますか?(クラス内が空のEntityは作成してみましたが、上記と同じエラーメッセージ)
  4. containが記載されていれば、joinは記載する必要はありますか?(記載してもしなくても上記と同じエラーメッセージ)

エラーの解決もとより、ファイルや記述は何が必要で何が不要か不明な状態です。

理解が浅くて申し訳ございませんが、お付き合いいただけると幸いです。

よろしくお願いします。

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

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

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

    クリップを取り消します

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

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

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

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

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

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

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

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

    質問の評価を下げる

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

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

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

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

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

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

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

    詳細な説明はこちら

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

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

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

回答 2

0

数百カラムをすべて select するんですか?
なんでそういうことしなきゃならないのか設計を見直すべきではないかとは思いますが……

さて CakePHP の ORM は、join したものをいきなり取り出したりしないようになっています。(select で明示的に指定していれば当然取り出しますが)
https://book.cakephp.org/3.0/ja/orm/retrieving-data-and-resultsets.html#contain

同時に取り出したければ contain で指定してください。

$this->test_1->find()
             ->join([
               ...
             ])
             ->where([
                 'id' => $id
             ])
             ->contain(['test_2', 'test_3'])
             ->first();

以下、追記部分への追加

test_2, test_3 とも、Entity, Table を構築しなければなりません。手作業で作成するのではなく、bin/cake bake を利用して、作らせるのがよいです。

投稿

編集

  • 回答の評価を上げる

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

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

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

  • 回答の評価を下げる

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

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

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

  • 2018/10/04 18:26

    エラー内容変わりました!「The test_2 association is not defined on test_1」。https://book.cakephp.org/3.0/ja/orm/associations.htmlのような感じで、テーブル同士の関係性を定義するってことですかね?

    キャンセル

  • 2018/10/04 18:41

    はい。ORM が関係性を認識できていない(ORM のデフォルト名称に合っていないから)からですね。

    キャンセル

  • 2018/10/05 11:24 編集

    ありがとうございます。試行錯誤したのですが、エラーが解決しないため、補足で追記に関し、ご回答頂けると幸いです。

    キャンセル

0

hasOne定義後のfindしているコードを載せないとなんとも言えません。

投稿

  • 回答の評価を上げる

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

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

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

  • 回答の評価を下げる

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

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

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

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

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

関連した質問

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