Отмена повторной отправки формы
Мне часто задают вопросы относительно отмены повторной отправки формы. Например, Вы сделали форму добавления комментария, добавили обработчик на эту же страницу. Затем при добавлении комментария он успешно добавляется, но стоит пользователю нажать F5, как форма будет отправлена ещё раз. А F5 пользователь может легко нажать, если страница будет долго грузиться. В итоге, вместо 1-го комментария будет целых 2, а то и больше. В этой статье я покажу, как этого можно избежать.
Для начала разберём более подробно проблему на примере этого кода:
<?php
if (!empty($_POST["sqr"])) {
echo "Квадрат числа ".$_POST["x"]." равен ".pow($_POST["x"], 2);
}
?>
<form name="form" action="" method="post">
<div>
<input type="text" name="x" />
<br />
<input type="submit" value="Возвести в квадрат" name="sqr" />
</div>
</form>
Нажав на кнопку "Возвести в квадрат", Вы увидите результат работы скрипта. Но стоит после этого пользователю нажать F5, как скрипт снова будет выполняться. В данном случае, это не так критично, как с добавлением комментарием, однако, зачем нужна лишняя нагрузка на сервер?
Теперь поговорим о способах решения данной проблемы. Первый способ - выделить скрипт обработки в отдельный файл. Тогда в атрибуте action у тега form надо добавить путь к этому скрипту. А сам скрипт должен сохранять куда-нибудь результат своих действий, либо переменные пришедшие на скрипт, а после делать редирект обратно. В общем, смотрите код скрипта:
<?php
session_start();
$_SESSION["x"] = $_POST["x"];
header("Location: ".$_SERVER["HTTP_REFERER"]);
exit;
?>
А код страницы с формой теперь будет выглядеть так:
<?php
session_start();
if (isset($_SESSION["x"])) echo "Квадрат числа ".$_SESSION["x"]." равен ".pow($_SESSION["x"], 2);
}
?>
<form name="form" action="request.php" method="post">
<div>
<input type="text" name="x" />
<br />
<input type="submit" value="Возвести в квадрат" name="sqr" />
</div>
</form>
Недостаток этого подхода очевиден - приходится создавать ещё один файл для такого простого скрипта. Поэтому рассказываю и про второй способ, как можно избежать повторной отправки формы:
<?php
session_start();
if (!empty($_POST["sqr"])) {
$_SESSION["x"] = $_POST["x"];
header("Location: ".$_SERVER["REQUEST_URI"]);
exit;
}
if (isset($_SESSION["x"])) echo "Квадрат числа ".$_SESSION["x"]." равен ".pow($_SESSION["x"], 2);
?>
<form name="form" action="" method="post">
<div>
<input type="text" name="x" />
<br />
<input type="submit" value="Возвести в квадрат" name="sqr" />
</div>
</form>
Здесь обработка снова происходит в этом же файле, но ключевое отличие - это наличие редиректа на эту же страницу в конце. В результате, страница перегрузится после отправки формы и браузер при нажатии F5 не будет предлагать пользователю отправить форму ещё раз.
Подведу итог того, как отменить повторную отправку формы:
- Либо делать обработку в отдельном файле, а затем оттуда делать редирект назад.
- Либо делать обработку в том же файле, что и форма, но при этом после обработки делать редирект на ту же страницу.
Вот так это делается в простых скриптах. Да и, в сложных, в конечном счёте делается то же самое.
-
- Михаил Русаков
Комментарии (12):
Или написать в форме action="."
Ответить
все это конечно понятно. Тоже хотел использовать способ на подобии, но у меня форма очень большая. Актуально ли писать значения формы из 57 полей в $_SESSION ?
Ответить
При таком примере писать придется. Редирект убьет все POSTы
Ответить
это и так понятно. Мой вопрос стоял актуально ли записывать такое значение полей 57 в $_SESSION
Ответить
А почему нет? <code> foreach($_POST as $k => $v) $_SESSION[$k] => $v; </code>
Ответить
А я делал по проще: в форму добавил скрытый input, в котором храниться случайное число генерирующееся php. А также при генерации оно забивается в сессии. Затем при принятии формы сначала проверяется совпадение числа, и если нет совпадения форма не обрабатывается. Это как скрытая капча, которая сама вводиться
Ответить
Я сделал так: if (isset($_POST['knopka'])) { echo 'Работает'; unset ($_POST['knopka']); }
Ответить
А меня интересует совсем противоположный вопрос! А как зделать чтоб отправка формы продублировалясь три раза с интервалом в секунду! Какой обрабочик нужно добавить в код?
Ответить
Здравствуйте. Не очень понял как делать. Изучаю Ваш курс основы PHP. Перепроверил код, но все равно при обновлении страницы комментарий дублируется. А у Вас на видео все нормально, без всяких редиректов. Вот сам код: <h2>Добавить запись</h2> <form name="guestbook" action="" method="post"> <table> <tr> <td>Имя:</td> <td> <input type="text" name="name" /> </td> </tr> <tr> <td>Комментарий:</td> <td> <input type="text" name="comment" /> </td> </tr> <tr> <td colspan="2"> <input type="submit" name="button_guestbook" value="Добавить" /> </td> </tr> </table> </form> <h2>Записи в гостевой книге:</h2> <div> <?php if (!empty($_POST["button_guestbook"])) { $name = htmlspecialchars($_POST["name"]); $comment = htmlspecialchars($_POST["comment"]); if ((strlen($name) < 3) || (strlen($comment) < 3)) $success = false; else $success = addGuestBookComment($name, $comment); if (!$success) { $alert = "Проверьте введёные данные"; include "alert.php"; } } $comments = getAllGuestBookComments(); for ($i = 0; $i < count($comments); $i++) { $name = $comments[$i]["name"]; $comment = $comments[$i]["comment"]; include "blocks/guestbook_comment.php"; } ?> </div>
Ответить
Добрый день. У меня не получается предотвратить повторную отправку. Выходят ошибки. Warning: session_start(): Cannot send session cookie - headers already sent by (output started at D:\webServer\OpenServer\domains\sample.local\index.php:3) in D:\webServer\OpenServer\domains\sample.local\index.php on line 30 Warning: session_start(): Cannot send session cache limiter - headers already sent (output started at D:\webServer\OpenServer\domains\sample.local\index.php:3) in D:\webServer\OpenServer\domains\sample.local\index.php on line 30 Warning: Cannot modify header information - headers already sent by (output started at D:\webServer\OpenServer\domains\sample.local\index.php:3) in D:\webServer\OpenServer\domains\sample.local\index.php on line 182
Ответить
Я конечно извиняюсь, но вы сами то запускали, тестили свои скрипты, во первых после нажатия и отправки запроса белый экран, результаты почему то выводятся в консоле, а во вторых это не работает, т.к. при обновлении данные отправляются повторно. Только что проверил!
Ответить
В статью необходимо сделать важное дополнение о том, что в основном файле сайта (index.php) после вывода результатов обработки формы необходимо обязательно закрыть сессию командой session_destroy(); , иначе скрипт обработки формы будет всегда выводить один и тот же результат вне зависимости от значений введенных в форму, т.к. он использует данные из сессии, а не из формы. Проверено на собственном скрипте.
Ответить
Для добавления комментариев надо войти в систему.
Если Вы ещё не зарегистрированы на сайте, то сначала зарегистрируйтесь.