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

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

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

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

Q&A

解決済

3回答

773閲覧

PHPにて掲示板を作成していますが、エラーの際に余計な文言が出力されてしまいます。

susumou

総合スコア2

PHP

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

0グッド

1クリップ

投稿2020/09/13 11:31

前提・実現したいこと

ここに質問の内容を詳しく書いてください。
(例)PHP(CakePHP)で●●なシステムを作っています。
■■な機能を実装中に以下のエラーメッセージが発生しました。

発生している問題・エラーメッセージ

プログラミング初心者です。PHPで掲示板を作成しているのですが、 ①nameとcommentを入力した際には、ブラウザにてどちらも出力(入力した時間も出力)→これはできています。 ②nameかcommentのいずれかだけを入力した場合はエラー→こちらはできています。 ③②でエラーが出た場合、投稿一覧には何も出力したくないのですが、「●: -2020-09-13 08:01:32」のように、出力されてしまいます。上記のような出力が出ないようにするためには、どのようにすれば良いでしょうか?ご教示お願い致しますm(__)m

該当のソースコード

<?php $filename = 'TokyoTyrantTable.csv'; $name=''; $comment=''; $name_max = 20; $comment_max = 100; $log = date('(-Y-m-d H:i:s)'); $fp = fopen($filename, 'a+b'); if ($_SERVER['REQUEST_METHOD'] === 'POST') { $name = htmlspecialchars($_POST['name']); $comment = htmlspecialchars($_POST['comment']); fputcsv($fp, [$name, $comment,$log]); rewind($fp); } while ($row = fgetcsv($fp)) { $rows[] = $row; } fclose($fp); ?> <!DOCTYPE html> <html lang="ja"> <head> <meta charset="UTF-8"> <link href="style.php" rel="stylesheet" type="text/css" media="all"> <title>掲示板</title> </head> <body> <h1>掲示板</h1> <section> <h2>新規投稿</h2> <?php if (mb_strlen($name) > $name_max){?> <p><?php print '名前は20文字以内で入力してください';?></p> <?php } ?> <?php if(mb_strlen($name) === 0){?> <p><?php print '名前を入力してください';?></p> <?php } ?> <?php if (mb_strlen($comment) > $comment_max){?> <p><?php print 'ひとことは100文字以内で入力してください';?></p> <?php } ?> <?php if(mb_strlen($comment) === 0){?> <p><?php print 'ひとことを入力してください';?></p> <?php } ?>
<form action="" method="post"> <div class="name"><span class="label">お名前:</span><input type="text" name="name" value=""></div> <div class="honbun"><span class="label">本文:</span><textarea name="comment" cols="30" rows="3" maxlength="80" wrap="hard" placeholder="100字以内で入力してください。"></textarea></div> <input type="submit" value="投稿"> </form>
</section> <section class="toukou"> <h2>投稿一覧</h2> <?php if (!empty($rows)){?> <?php foreach ($rows as $row){ ?> <?php if((empty($name))&&(empty($comment))) { ?> <li><?php print $row[0].":"?> <?php print $row[1] ?> <?php print $row[2] ?></li> <?php } else{?> <?php break; } ?> <?php } ?> <?php } ?> </body> </html>
PHP

試したこと

試している箇所を抜粋いたします。

<h2>投稿一覧</h2> <?php if (!empty($rows)){?> <?php foreach ($rows as $row){ ?> <?php if((empty($name))&&(empty($comment))) { ?> <li><?php print $row[0].":"?> <?php print $row[1] ?> <?php print $row[2] ?></li> <?php } else{?> <?php break; } ?> <?php } ?> <?php } ?>

補足情報(FW/ツールのバージョンなど)

ここにより詳細な情報を記載してください。

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

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

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

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

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

guest

回答3

0

おかしなところがあります。

php

1$fp = fopen($filename, 'a+b'); 2if ($_SERVER['REQUEST_METHOD'] === 'POST') { 3$name = htmlspecialchars($_POST['name']); 4$comment = htmlspecialchars($_POST['comment']); 5fputcsv($fp, [$name, $comment,$log]); 6rewind($fp); 7}

これ、POST受信したら$name$commentがどういう状態でも
fputcsv()にて出力しちゃいますが、それでいいのでしょうか?

fputcsv()には、
区切り文字の指定や、フィールドを囲む記号の指定や、特殊な文字をエスケープする処理の指定など、
便利な機能があるので、有効に使うべきです。

php

1<h2>投稿一覧</h2> 2<?php if (!empty($rows)){?> 3<?php foreach ($rows as $row){ ?> 4<?php if((empty($name))&&(empty($comment))) { ?> 5<li><?php print $row[0].":"?> <?php print $row[1] ?> <?php print $row[2] ?></li> 6<?php } else{?> 7<?php break; } ?> 8<?php } ?> 9<?php } ?>

$rowsから一つずつ$rowに引っ張り出して、
$rowと直接関係のない$name$commentで評価してから$rowを出力するけど、
この時点で$rowはたぶん一行丸々のカンマ区切り文字列のままになってて
$row[0]とか$row[1]とかでアクセスすると左端から0番目や1番目の文字を出力するような感じにならないでしょうか。

それと、直接の問題ではないですが、
一行ずつPHPの開始タグと終了タグを書くと、大変読みにくくミスの発見も遅れますし、動作も遅いようです。
アルゴリズムのおかしいところを目をつむり先の箇所を書き直すと、

php

1<h2>投稿一覧</h2> 2<?php 3 if (!empty($rows)) { 4 foreach ($rows as $row) { 5 if ((empty($name)) && (empty($comment))) { 6?> 7<li><?php print $row[0].":"; ?> <?php print $row[1]; ?> <?php print $row[2]; ?></li> 8<?php 9 } 10 else { 11 break; 12 } 13 } 14 } 15?>

などとして、インデントレベルを適切にしてアルゴリズムがおかしいところがないかを追いやすくしたほうが良いかと。

あと、この先遭遇するであろう問題点も書き出しておきます。

●名前やコメントに「,」を含めてしまうと、CSVとしての文字区切りと混同してデータ処理がおかしくなる。(htmlspecialchars()は「,」の処理をしない。)
●コメントの入力でmaxlength=80としたら81文字以上99文字以下を受け付けられないのでは? 重要な数字は定数を駆使して一箇所直すだけで全部反映されるようにする。
●データファイルにhtmlspecialchars()で加工したあとの文字列を記録すると、検索機能の実装に支障が出るので、ファイルへの記録は加工前の素の文字列で、htmlへの出力直前に加工して出力するのを徹底する。
●追記のみのようだけど、コメントの編集や削除機能がちゃんとできるのかどうか。

フォームで入力する名前やコメントに「,」が含まれるとまず破綻するので、
この手の機能を作るときはデータの区切り文字列として例えば「<>」を使ったりするけど、
フォームで入力する名前やコメントに「<>」が含まれたら結局同じなので、
フォーム入力できない文字(たとえばTABコード)にするか、
もしくはフォーム入力された文字列に「<>」が含まれていたら可逆性のある文字列に置き換える方法もあります。
以上蛇足。

投稿2020/09/14 02:04

編集2020/09/14 02:11
退会済みユーザー

退会済みユーザー

総合スコア0

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

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

susumou

2020/09/15 01:11

具体的なアドバイスありがとうございますm(__)m 初心者ですので、非常に助かります。 前半部分(fputcsv部分)は無事解決できましたので、後半部分(新規投稿以下)も修正・追記していこうと思います。 ありがとうございました
guest

0

そもそもの部分なのですが、DBを使わずにテキストファイルのようなもので管理する際にはそのような判定する処理が多少手間がかかるので、nameとcommentの入力を必須化すれば良いのではないでしょうか。

投稿2020/09/14 06:06

kai0310

総合スコア2070

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

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

susumou

2020/09/15 01:12

ありがとうございます!必須化することで解決できました!
guest

0

ベストアンサー

fputcsv($fp, [$name, $comment,$log]);

を、

if (!empty($name) and !empty($comment)) {
}

で囲ってあげれば良いと思います。

投稿2020/09/14 02:03

退会済みユーザー

退会済みユーザー

総合スコア0

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

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

susumou

2020/09/15 01:06

簡潔にご回答いただきありがとうございます。アドバイスいただいた箇所と後半のforeach部分を少し修正することで解決いたしました。ありがとうございます。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.48%

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

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

質問する

関連した質問