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

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

新規登録して質問してみよう
ただいま回答率
85.35%
Laravel

LaravelとはTaylor Otwellによって開発された、オープンソースなPHPフレームワークです。Laravelはシンプルで表現的なシンタックスを持ち合わせており、ウェブアプリケーション開発の手助けをしてくれます。

PHP

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

Q&A

解決済

2回答

1456閲覧

[Laravel] Eloquentの最適化についての質問

yuuuta

総合スコア1

Laravel

LaravelとはTaylor Otwellによって開発された、オープンソースなPHPフレームワークです。Laravelはシンプルで表現的なシンタックスを持ち合わせており、ウェブアプリケーション開発の手助けをしてくれます。

PHP

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

0グッド

0クリップ

投稿2020/10/02 06:35

[やりたいこと]
larabelで集計ツールを作っています。

[問題]
集計結果の表示はできるもののページの処理に時間がかかります。デバックツールで表示されているクエリ数が1030となることからクエリ数が多すぎると思うのですが、クエリ数を減らす書き方がわからず困っています。
特にforeachの中でwhereInメソッドを入れてから重くなりました。
テーブル関係はt_staffがt_reportをhas_many, t_reportがt_report_detailをhas_manyしています。

class SalesTotalController extends Controller { /** * 営業集計 初期表示 * @param Request $request [description] */ public function Index(Request $request) { $staffs = Tstaff::orderBy('staff_code', 'asc')->get(); //加算日(t_common/common_id=max_summary_spanの日数(value1)をプラスした日付) $add_day = Tcommon::where('common_id', 'max_summary_span')->first(['value1'])['value1']; foreach ( $staffs as $staff ) { $staff_code = $staff->staff_code; if ( $request->start_date && $request->end_date) { $start_date = new Carbon($request->start_date); $end_date = new Carbon($request->end_date); } else { //システム開始日 $start_date = new Carbon('2020-08-01'); //終了日 $end_date = $start_date->copy()->addDay($add_day); } //開始日と終了日の差 $diff = ($end_date->timestamp - $start_date->timestamp) / (60 * 60 * 24) + 1; //報告書数 $t_report_sum[$staff_code] = Treport::where('staff_code', $staff_code) ->where('valid_flag', 1) ->where('status_flag', 1) ->where('submitted_datetime', '>=', $start_date ) ->where('submitted_datetime', '<=', $end_date ) ->count(); //要望書数 $t_proposal_sum[$staff_code] = Tproposal::where('staff_code', $staff_code) ->where('valid_flag', 1) ->where('status_flag', 1) ->where('submitted_datetime', '>=', $start_date ) ->where('submitted_datetime', '<=', $end_date ) ->count(); $t_report_ids = Treport::where('staff_code', $staff_code) ->where('valid_flag', 1) ->where('status_flag', 1) ->where('submitted_datetime', '>=', $start_date ) ->where('submitted_datetime', '<=', $end_date ) ->get(['report_number']) ->toArray(); //ここらへんから重くなる $t_report_detail = Treportdetail::whereIn('report_number', $t_report_ids); //営業校数(t_report_detailの合計) $t_report_detail_sum[$staff_code] = $t_report_detail->count(); //新規数(記明細レコード件数からreport1(テキスト)がnullまたはブランクのレコードを除外した件数合計) $t_report_detail_new_sum[$staff_code] = Treportdetail::whereIn('report_number', $t_report_ids) ->where('report1', '!=', '') ->count(); //継続数(上記明細レコード件数からreport2(テキスト)がnullまたはブランクのレコードを除外した件数合計) $t_report_detail_existing_sum[$staff_code] = Treportdetail::whereIn('report_number', $t_report_ids) ->where('report2', '!=', '') ->count(); //入校(上記明細レコード件数の内、action_type=1の件数合計) $t_report_detail_meeting_sum[$staff_code] = Treportdetail::whereIn('report_number', $t_report_ids) ->where('action_type', 1) ->count(); //事務(上記明細レコード件数の内、action_type=2の件数合計) $t_report_detail_office_sum[$staff_code] = Treportdetail::whereIn('report_number', $t_report_ids) ->where('action_type', 2) ->count(); //アポ(上記明細レコード件数の内、action_type=3の件数合計) $t_report_detail_appointment_sum[$staff_code] = Treportdetail::whereIn('report_number', $t_report_ids) ->where('action_type', 3) ->count(); //電話(上記明細レコード件数の内、action_type=4の件数合計) $t_report_detail_tel_sum[$staff_code] = Treportdetail::whereIn('report_number', $t_report_ids) ->where('action_type', 4) ->count(); //メール(上記明細レコード件数の内、action_type=5の件数合計) $t_report_detail_mail_sum[$staff_code] = Treportdetail::whereIn('report_number', $t_report_ids) ->where('action_type', 5) ->count(); //週ごとの校数 $week = 1; while($diff > 0) { $week_start_date = $start_date->copy()->addDay(($week - 1) * 7); if ($diff < 7) { $week_end_date = $end_date; } else { $week_end_date = $start_date->copy()->addDay($week * 7 - 1); } $t_report_detail_week_sum[$staff_code][$week] = Treportdetail::whereIn('report_number', $t_report_ids) ->where('created_datetime', '>=', $week_start_date ) ->where('created_datetime', '<=', $week_end_date ) ->count(); $diff = $diff - 7; $week++; } } //検索結果をsessionで保持 $request->session()->flash('start_date', $request->start_date); $request->session()->flash('end_date', $request->end_date); $data = [ 'staff_name' => $request->staff_name, 'branch_name' => $request->branch_name, 'authority_flag' => $request->authority_flag, 'staffs' => $staffs, 't_report_sum' => $t_report_sum, 't_proposal_sum' => $t_proposal_sum, 't_report_detail_sum' => $t_report_detail_sum, 't_report_detail_new_sum' => $t_report_detail_new_sum, 't_report_detail_existing_sum' => $t_report_detail_existing_sum, 't_report_detail_meeting_sum' => $t_report_detail_meeting_sum, 't_report_detail_office_sum' => $t_report_detail_office_sum, 't_report_detail_appointment_sum' => $t_report_detail_appointment_sum, 't_report_detail_tel_sum' => $t_report_detail_tel_sum, 't_report_detail_mail_sum' => $t_report_detail_mail_sum, 'start_date' => $start_date, 'end_date' => $end_date, 't_report_detail_week_sum' => $t_report_detail_week_sum, ]; return view('totalsales_result', $data); } }

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

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

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

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

phper.k

2020/10/02 06:45 編集

テーブルの定義、サンプルデータがないと回答は困難です。 その前に、リレーション、および Eager loading について調査してください。
guest

回答2

0

ベストアンサー

パッと見、問題になってそうなのはt_report.report_numberです。
たぶんt_reportとt_report_detailレコードは大量にあるのでしょう。
IN句でフルスキャンになっているのでは?
また、report1、report2の否定を取っていますが、これもインデックス効いていないはずです。

すぐ思いつく対策は3つ。

  1. JOINする
  2. EXISTS句を使う
  3. あらかじめ日次(月次)で集計しておく

3はちょっと後ろ向きですが、集計テーブル作って日次か月次バッチ動かすだけなので簡単です。

投稿2020/10/03 05:44

kgm

総合スコア275

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

0

解決したのならば、どう対応したのか書いてもらえませんか?
同じところで躓いた人の参考になるはずです。

投稿2020/10/04 13:26

kgm

総合スコア275

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.35%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問