Прервать Ajax-запросы, используя jQuery
С помощью jQuery, как я могу отменить / прервать Ajax-запрос, от которого я еще не получил ответ?
19 ответов
Большинство методов jQuery Ajax возвращают объект XMLHttpRequest (или эквивалентный), поэтому вы можете просто использовать abort()
,
Смотрите документацию:
- Прервать метод ( MSDN). Отменяет текущий HTTP-запрос.
- прервать () ( MDN). Если запрос уже отправлен, этот метод прервет запрос.
var xhr = $.ajax({
type: "POST",
url: "some.php",
data: "name=John&location=Boston",
success: function(msg){
alert( "Data Saved: " + msg );
}
});
//kill the request
xhr.abort()
ОБНОВЛЕНИЕ: Начиная с jQuery 1.5 возвращаемый объект является оберткой для собственного объекта XMLHttpRequest, называемого jqXHR. Этот объект, по-видимому, предоставляет все собственные свойства и методы, поэтому приведенный выше пример все еще работает. См . Объект jqXHR (документация по jQuery API).
ОБНОВЛЕНИЕ 2: Начиная с jQuery 3, метод ajax теперь возвращает обещание с дополнительными методами (например, abort), поэтому приведенный выше код все еще работает, хотя возвращаемый объект не является xhr
больше Смотрите блог 3.0 здесь.
ОБНОВЛЕНИЕ 3: xhr.abort()
по-прежнему работает на jQuery 3.x. Не думайте, что обновление 2 корректно. Больше информации о репозитории jQuery Github.
Вы не можете отозвать запрос, но вы можете установить значение тайм-аута, после которого ответ будет игнорироваться. Смотрите эту страницу для JQuery AJAX вариантов. Я полагаю, что ваш обратный вызов ошибки будет вызван, если превышен период ожидания. Уже есть время ожидания по умолчанию для каждого запроса AJAX.
Вы также можете использовать метод abort() для объекта запроса, но, хотя это приведет к тому, что клиент прекратит прослушивать событие, возможно, он не остановит его обработку сервером.
Это асинхронный запрос, то есть после отправки он уже есть.
Если ваш сервер запускает очень дорогую операцию из-за запроса AJAX, лучшее, что вы можете сделать, - это открыть свой сервер для прослушивания запросов отмены и отправить отдельный запрос AJAX, уведомляющий сервер о необходимости прекратить все, что он делает.
В противном случае просто игнорируйте ответ AJAX.
Сохраните вызовы, которые вы делаете, в массиве, а затем вызовите xhr.abort() для каждого.
ОГРОМНАЯ ПРОСМОТР: Вы можете прервать запрос, но это только на стороне клиента. Серверная сторона все еще может обрабатывать запрос. Если вы используете что-то вроде PHP или ASP с данными сеанса, данные сеанса блокируются до завершения Ajax. Таким образом, чтобы позволить пользователю продолжить просмотр сайта, вы должны вызвать session_write_close(). Это сохранит сеанс и разблокирует его, так что другие страницы, ожидающие продолжения, продолжатся. Без этого несколько страниц могут ожидать снятия блокировки.
Запросы AJAX могут не выполняться в том порядке, в котором они были запущены. Вместо прерывания вы можете игнорировать все ответы AJAX, за исключением самого последнего:
- Создать счетчик
- Увеличьте счетчик при запуске запроса AJAX
- Использовать текущее значение счетчика для "печати" запроса
- В случае успешного обратного вызова сравните штамп со счетчиком, чтобы проверить, был ли это последний запрос
Грубый набросок кода:
var xhrCount = 0;
function sendXHR() {
// sequence number for the current invocation of function
var seqNumber = ++xhrCount;
$.post("/echo/json/", { delay: Math.floor(Math.random() * 5) }, function() {
// this works because of the way closures work
if (seqNumber === xhrCount) {
console.log("Process the response");
} else {
console.log("Ignore the response");
}
});
}
sendXHR();
sendXHR();
sendXHR();
// AJAX requests complete in any order but only the last
// one will trigger "Process the response" message
Всегда лучше делать что-то подобное.
var $request;
if ($request != null){
$request.abort();
$request = null;
}
$request = $.ajax({
type : "POST", //TODO: Must be changed to POST
url : "yourfile.php",
data : "data"
}).done(function(msg) {
alert(msg);
});
Но гораздо лучше, если вы проверите оператор if, чтобы проверить, является ли запрос ajax нулевым или нет.
Нам просто нужно было обойти эту проблему и протестировать три разных подхода.
- отменяет запрос, предложенный @meouw
- выполнить весь запрос, но обрабатывает только результат последней отправки
- предотвращает новые запросы, пока еще один ожидает
var Ajax1 = {
call: function() {
if (typeof this.xhr !== 'undefined')
this.xhr.abort();
this.xhr = $.ajax({
url: 'your/long/running/request/path',
type: 'GET',
success: function(data) {
//process response
}
});
}
};
var Ajax2 = {
counter: 0,
call: function() {
var self = this,
seq = ++this.counter;
$.ajax({
url: 'your/long/running/request/path',
type: 'GET',
success: function(data) {
if (seq === self.counter) {
//process response
}
}
});
}
};
var Ajax3 = {
active: false,
call: function() {
if (this.active === false) {
this.active = true;
var self = this;
$.ajax({
url: 'your/long/running/request/path',
type: 'GET',
success: function(data) {
//process response
},
complete: function() {
self.active = false;
}
});
}
}
};
$(function() {
$('#button').click(function(e) {
Ajax3.call();
});
})
<script src="https://ajax.googleapis.com/ajax/libs/jquery/2.1.1/jquery.min.js"></script>
<input id="button" type="button" value="click" />
В нашем случае мы решили использовать подход № 3, поскольку он создает меньшую нагрузку на сервер. Но я не уверен на 100%, если jQuery гарантирует вызов метода.complete(), это может привести к тупиковой ситуации. В наших тестах мы не могли воспроизвести такую ситуацию.
Решение meouw верное, но если вы заинтересованы в большем контроле, вы можете попробовать плагин Ajax Manager для jQuery.
Просто позвоните xhr.abort (), будь то объект jquery ajax или собственный объект XMLHTTPRequest.
пример:
//jQuery ajax
$(document).ready(function(){
var xhr = $.get('/server');
setTimeout(function(){xhr.abort();}, 2000);
});
//native XMLHTTPRequest
var xhr = new XMLHttpRequest();
xhr.open('GET','/server',true);
xhr.send();
setTimeout(function(){xhr.abort();}, 2000);
Вы можете прервать любой непрерывный вызов AJAX, используя эту
<input id="searchbox" name="searchbox" type="text" />
<script src="http://code.jquery.com/jquery-1.11.0.min.js"></script>
<script type="text/javascript">
var request = null;
$('#searchbox').keyup(function () {
var id = $(this).val();
request = $.ajax({
type: "POST", //TODO: Must be changed to POST
url: "index.php",
data: {'id':id},
success: function () {
},
beforeSend: function () {
if (request !== null) {
request.abort();
}
}
});
});
</script>
Как отметили многие в потоке, только потому, что запрос отменен на стороне клиента, сервер все равно будет обрабатывать запрос. Это создает ненужную нагрузку на сервер, потому что он выполняет работу, которую мы прекратили слушать на внешнем интерфейсе.
Проблема, которую я пытался решить (к которой могут также натолкнуться другие), заключается в том, что когда пользователь вводил информацию в поле ввода, я хотел отменить запрос о типе чувствительности Google Instant.
Чтобы не запускать ненужные запросы и поддерживать работоспособность внешнего интерфейса, я сделал следующее:
var xhrQueue = [];
var xhrCount = 0;
$('#search_q').keyup(function(){
xhrQueue.push(xhrCount);
setTimeout(function(){
xhrCount = ++xhrCount;
if (xhrCount === xhrQueue.length) {
// Fire Your XHR //
}
}, 150);
});
По сути, это будет отправлять один запрос каждые 150 мс (переменная, которую вы можете настроить для своих нужд). Если у вас возникли проблемы с пониманием, что именно здесь происходит, войдите xhrCount
а также xhrQueue
на консоль непосредственно перед блоком if.
Я занимался поиском в реальном времени и мне нужно было отменить ожидающие запросы, которые могли занять больше времени, чем последний / самый последний запрос.
В моем случае я использовал что-то вроде этого:
//On document ready
var ajax_inprocess = false;
$(document).ajaxStart(function() {
ajax_inprocess = true;
});
$(document).ajaxStop(function() {
ajax_inprocess = false;
});
//Snippet from live search function
if (ajax_inprocess == true)
{
request.abort();
}
//Call for new request
Просто используйте ajax.abort(), например, вы можете прервать любой ожидающий запрос ajax перед отправкой другого, подобного этому.
//check for existing ajax request
if(ajax){
ajax.abort();
}
//then you make another ajax request
$.ajax(
//your code here
);
Нет надежного способа сделать это, и я бы даже не попробовал его, как только запрос на ходу; единственный способ реагировать разумно - игнорировать ответ.
в большинстве случаев это может произойти в таких ситуациях, как: пользователь слишком часто нажимает на кнопку, вызывая многократный XHR, здесь у вас есть много опций: либо блокировать кнопку до возвращения XHR, либо даже не запускать новый XHR, пока другой выполняет подсказку пользователь может откинуться назад или отказаться от любого ожидающего ответа XHR, кроме недавнего.
Следующий код показывает как инициирование, так и прерывание Ajax-запроса:
function libAjax(){
var req;
function start(){
req = $.ajax({
url: '1.php',
success: function(data){
console.log(data)
}
});
}
function stop(){
req.abort();
}
return {start:start,stop:stop}
}
var obj = libAjax();
$(".go").click(function(){
obj.start();
})
$(".stop").click(function(){
obj.stop();
})
<script src="https://ajax.googleapis.com/ajax/libs/jquery/2.1.1/jquery.min.js"></script>
<input type="button" class="go" value="GO!" >
<input type="button" class="stop" value="STOP!" >
У меня была проблема с опросом, и как только страница была закрыта, опрос продолжался, поэтому в моем случае пользователь пропустил обновление, так как значение mysql устанавливалось в течение следующих 50 секунд после закрытия страницы, даже если я убил запрос ajax, я обошлось, использование $_SESSION для установки переменной не будет обновляться в самом опросе до тех пор, пока он не закончится и не запустится новый, поэтому в моей базе данных было установлено значение 0 = offpage, в то время как я при опросе я запрашиваю эту строку и возвращаю false; когда он равен 0, запрос при опросе, очевидно, даст вам текущие значения...
Надеюсь это помогло
Если xhr.abort();
вызывает перезагрузку страницы,
Тогда вы можете установить onreadystatechange
перед прерыванием, чтобы предотвратить:
// ↓ prevent page reload by abort()
xhr.onreadystatechange = null;
// ↓ may cause page reload
xhr.abort();
Я поделился демонстрацией, которая демонстрирует, как отменить запрос AJAX - если данные не возвращаются с сервера в течение предварительно определенного времени ожидания.
HTML:
<div id="info"></div>
JS CODE:
var isDataReceived= false, waitTime= 1000;
$(function() {
// Ajax request sent.
var xhr= $.ajax({
url: 'http://api.joind.in/v2.1/talks/10889',
data: {
format: 'json'
},
dataType: 'jsonp',
success: function(data) {
isDataReceived= true;
$('#info').text(data.talks[0].talk_title);
},
type: 'GET'
});
// Cancel ajax request if data is not loaded within 1sec.
setTimeout(function(){
if(!isDataReceived)
xhr.abort();
},waitTime);
});
Это моя реализация, основанная на многих ответах выше:
var activeRequest = false; //global var
var filters = {...};
apply_filters(filters);
//function triggering the ajax request
function apply_filters(filters){
//prepare data and other functionalities
var data = {};
//limit the ajax calls
if (activeRequest === false){
activeRequest = true;
}else{
//abort if another ajax call is pending
$request.abort();
//just to be sure the ajax didn't complete before and activeRequest it's already false
activeRequest = true;
}
$request = $.ajax({
url : window.location.origin + '/your-url.php',
data: data,
type:'POST',
beforeSend: function(){
$('#ajax-loader-custom').show();
$('#blur-on-loading').addClass('blur');
},
success:function(data_filters){
data_filters = $.parseJSON(data_filters);
if( data_filters.posts ) {
$(document).find('#multiple-products ul.products li:last-child').after(data_filters.posts).fadeIn();
}
else{
return;
}
$('#ajax-loader-custom').fadeOut();
},
complete: function() {
activeRequest = false;
}
});
}