- ベストアンサー
MySQL 検索結果のファイル保存と行数値入手
- MySQLで検索結果をファイル保存し、行数値を取得したい
- PHPを使用してMySQLデータベースに接続し、テーブルの作成とデータの追加には成功している
- しかし、検索結果をファイルに保存する際に行数値を取得することができず、エラーメッセージが表示される
- みんなの回答 (6)
- 専門家の回答
質問者が選んだベストアンサー
「SELECT * FROM my_database」 だとこのテーブルのデータがまるごと入ったmysqli_result オブジェクトを返すので mysqli_num_rows()に渡して行数のカウントができるが 「SELECT * FROM my_database INTO OUTFILE "output.csv"」 だとテーブルの内容を外部ファイルに取り出すということを実行し その結果が成功ならtrue・失敗ならfalseの区別がbool型が返るだけなので 「bool型の変数はmysqli_num_rows() の引数に使えないよ」 というエラーが出るということしょう。 ”MySQL での検索結果をファイル保存し、かつ検索データ数(行数値)を入手” を同時に行えるSQLがあればよいのでしょうけど、 同時に行わせることに強い理由がないのであれば それぞれを別々に行っても問題ないのではないでしょうか。 なお、データ数のカウントのほうは カウントするだけのために 全データを取り出してから、それを改めてカウントするのは無駄なので 「SELECT count(*) as XXX from テーブル名 (~where等)」 とやってDB上でカウントさせた値を直接取り出したほうが 効率的だと思わます。
その他の回答 (5)
- superside0
- ベストアンサー率64% (463/719)
補足に 「行数値や出力ファイル内容(の必要部分)を php 経由で html 側に渡す必要があります」 とかいてありますが、最終目的は 外部ファイル化することでなく HTMLをブラウザに渡すことなんですか? もしそういうことなら、そもそも外部ファイル化する必要もなくて $resの内容を HTMLにして出力すればよいだけの ことではないですか?
補足
本日は外出していて、レスポンスが遅れて申し訳ありませんでした。 >最終目的は 外部ファイル化することでなく >HTMLをブラウザに渡すことなんですか? >もしそういうことなら、そもそも外部ファイル化する必要もなくて >$resの内容を HTMLにして出力すればよいだけの >ことではないですか? 言葉足らずで申し訳ありませんでした。 最終的には、検索結果から必要情報を表示させ、見やすくしたものをブラウザで表示させ、それを Excel 型式のファイルで保存する事です。 ただし、ブラウザの表示内容を Excel に保存する部分は既に作成済みなので、「今回作業の最終目的は 検索結果をHTML(ブラウザ)に渡すこと」という表現をしてしまいました。 ご提示頂いた2つの方法の動作を確認いたしました。 今回、文字列の内容によって戻り値の型式が違う、という問題がメインだったので特に報告はしていませんでしたが、実際は INTO OUTFILE (ファイル名) FIELDS TERMINATED BY "," というように、カンマで区切る型式にしていました。 第1の方法(exec 文でリダイレクト)では、カンマ型式にならなかったので(多分、設定はできない?)、第2の方法の、ご提示頂いた php スプリクトを使わせて頂きました。 どちらも属性 644 のファイルが正常に作成できました。 これで、今回の問題は全て解決できました。 ありがとうございました。
- superside0
- ベストアンサー率64% (463/719)
> mysql ver 8.0.17 からUMASK の値が変更されて制限強化されている なるほど、もはやinto outfileを使うのはシステム管理の用途のみで アプリケーション用に使うべきではないという位置づけなのでしょうね。 ファイル化するのに SQLの select ~into outfile の機能を使う以外に 以下の2通りがあるので 別の方法を試してみてはどうでしょう。 ・PHPのシステムコール(OSのコマンドライン機能の呼び出し)で mysqlコマンド の出力をリダイレクトしてファイル化 例) exec ("mysql -u ユーザー -pパスワード DB名 -e 'SELECT * FROM my_database;' > /パス/ファイル"); ・PHPで $res = mysqli_query($link,"SELECT * FROM my_database"); でとりだした結果をそのままPHPの処理でファイルに書き出す 例) if($fp = fopen(ファイル名,'w')) { while ($row = mysqli_fetch_assoc($res)) { //日本語のコード変換が必要ならmb_convert_variables('SJIS-win', 'UTF-8', $row);みたいにして fputcsv($fp, $row); } fclose($fp); } else { 書き込みできないというエラーメッセージ処理 }
- superside0
- ベストアンサー率64% (463/719)
> まず、mysqli では cnt というインデックスは使えないようでエラーになります。 例で書いたSQL文の $sql = "SELECT count(*) as cnt FROM my_database"; に注目してほしいのですが、 「as cnt 」 ってつけてますよね。 なので、$row["count(*)"] でなく $row["cnt"] で 4が とりだせます。
補足
すみませんでした。 as cnt があるのを見逃していました。ご教示頂いたように実行すると 4 が取り出せました。 おかげさまで今回の問題はこれで解決したのですが、別件で1つ大きな問題に気がつきました。 superside0 様は、mysql には大変お詳しいようなので、便乗して質問させて頂く事お許し下さい。 問題というのは、「SELECT * FROM my_database INTO OUTFILE ・・・・」で作成されたファイルの属性が(想定した 644 ではなく) 640 になっている事で、これだと、他の PHP モジュールに移行した時に、このファイルが読み出せず、「Permission denied 。。。」というエラーになって しまいます。 ネット情報だと、mysql ver 8.0.17 からUMASK の値が変更されて制限強化されているようです。 この出力ファイルの所有者が mysql:mysql になっているためか、PHP で chmod をやっても失敗します。 本件に関し、何か対策はあるのでしょうか?
- superside0
- ベストアンサー率64% (463/719)
あ、すみません 先のは $row = mysql_fetch_assoc($res); じゃなくて $row = mysqli_fetch_assoc($res); でした。
- superside0
- ベストアンサー率64% (463/719)
> $row_count = mysqli_num_rows($res); >実際にやってみると、 $row_count は(4ではなく)1 となってしまいます。 mysqli_num_rows()は、行数を返す関数ですから SQLの結果として4 というデータが1行あるのであれば、当然1ですね。 DB側でカウントしたほうが無駄がないので PHP側でmysqli_num_rows()でカウントするのではなく $sql = "SELECT count(*) as cnt FROM my_database"; $res = mysqli_query($link, $sql); $row = mysql_fetch_assoc($res); print $row["cnt"]; でどうですか?
補足
コメントありがとうございました。実行内容をご報告します。 $res = mysqli_query($link, $sql); $row = mysqli_fetch_assoc($res); print $row["cnt"]; まず、mysqli では cnt というインデックスは使えないようでエラーになります。代わりに count 関数が使えるようです。(ただし私はこの関数を良く理解できていません) $row という連想配列を print_r($row); で確認すると、Array ( [COUNT(*)] => 4 ) が返ってくるのですが、count($row) の値を調べると 4 ではなく 1 となってしまいます。 初歩的な質問で恐縮ですが、4 を取り出す方法をご教示頂けたら幸いです。
お礼
これで、今回の問題は全て解決できました。 ありがとうございました。
補足
早速のコメント、深く感謝いたします。 文字列が「SELECT * FROM my_database」の場合と「SELECT * FROM my_database INTO OUTFILE "output.csv"」の場合で 戻り値の型式が全く違う、という事は私の想定外だったので、袋小路に入っていました。 ご説明、ありがとうございます。 ご教示頂いたように、「SELECT count(*) FROM my_database」という文字列を使って、データが4個だけの簡単なデータベースを作って動作確認しました。 mysql> SELECT COUNT(*) FROM my_database; +----------+ | COUNT(*) | +----------+ | 4 | +----------+ 1 row in set (0.00 sec) というように、4個のデータが入っている事を確認できたのですが、前回と同様にこの文字列を使って $res = mysqli_query($link, $sql); $row_count = mysqli_num_rows($res); 実際にやってみると、 $row_count は(4ではなく)1 となってしまいます。 この 4 の値を得る方法(関数)はあるのでしょうか? (追伸) 今回の質問の背景は、自宅サーバーにある MySQL を利用するユーザーは、PC のブラウザで操作する事を想定しています。 従って、行数値や出力ファイル内容(の必要部分)を php 経由で html 側に渡す必要があります。