Ответ 1
Если вы используете сеансы в функциях обработки AJAX, вы столкнетесь с проблемой, когда данные сеанса диска заблокированы первым запросом, поэтому каждый последующий запрос заканчивается ожиданием доступности данных сеанса до его продолжения. По сути, это заставляет асинхронные вызовы блокировать друг друга, вы получаете линейные ответы на запросы в хронологическом порядке - синхронные. (здесь справочная статья)
Решение состоит в том, чтобы использовать session_write_close
(docs), чтобы закрыть сеанс, как только он вам больше не понадобится. Это позволяет выполнять другие последующие запросы, поскольку данные сеанса будут "разблокированы".
Это, однако, тоже может ввести в заблуждение. Если вы вызываете session_write_close
прямо перед тем, как будете возвращать ответ, то вы не будете делать никаких выгод, потому что сеанс был бы разблокирован, как только был отправлен ответ. Таким образом, его нужно называть как можно раньше. Если вы используете архитектуру стиля post-back для запроса AJAX, это не так уж плохо, но если у вас большая структура, и ваш обработчик запросов является лишь ее частью, вам придется исследовать более высокоуровневое решение к неблокирующему использованию сеанса, поэтому ваши подкомпоненты не закрывают сеанс, ожидающий рамки, все еще открыт.
Один маршрут - это сеанс базы данных. Есть плюсы и минусы этого решения, которые выходят за рамки этого ответа - проверьте Google на исчерпывающее обсуждение. Другой маршрут - использовать функцию, которая открывает сеанс, добавляет переменную, а затем закрывает ее. Вы рискуете условиями гонки с этим решением, но здесь грубая схема:
function get_session_var($key, $default=null) {
if (strlen($key) < 1)
return null;
if (!isset($_SESSION) || !is_array($_SESSION)) {
session_start();
session_write_close();
}
if (array_key_exists($key, $_SESSION))
return $_SESSION[$key];
return $default;
}
function set_session_var($key, $value=null) {
if (strlen($key) < 1)
return false;
if ($value === null && array_key_exists($key, $_SESSION)) {
session_start();
unset($_SESSION[$key]);
} elseif ($value != null) {
session_start();
$_SESSION[$key] = $value;
} else {
return false;
}
session_write_close();
return true;
}