Использование XMLHttpRequest
BaselineWidely available *
This feature is well established and works across many devices and browser versions. It’s been available across browsers since July 2015.
* Some parts of this feature may have varying levels of support.
Это инструкция по использованиюXMLHttpRequest
для обмена информацией между сайтом и сервером поHTTP-протоколу.
Мы разберём как частые примеры использованияXMLHttpRequest
, так и более редкие.
Для отправки HTTP-запроса нужно создатьXMLHttpRequest
-объект, указать URL и отправить запрос. В результате запроса мы получим от сервера объект с подробной информацией, вроде тела ответа иHTTP-статуса.
function reqListener() { console.log(this.responseText);}const req = new XMLHttpRequest();req.addEventListener("load", reqListener);req.open("GET", "http://www.example.org/example.txt");req.send();
Типы запросов
Запрос, сделанный посредствомXMLHttpRequest
, может загружать данные синхронно или асинхронно. Тип запроса определяется опциональнымasync
аргументом (третий по счёту) методаXMLHttpRequest.open()
. Если он равенtrue
или не задан, запрос выполнится асинхронно, в противном случае — синхронно.
В чем разница между двумя типами запросов, а так же примеры использования, вы можете найтив статье с подробным описанием синхронных и асинхронных запросов. По возможности избегайте синхронных запросов, они блокируют взаимодействие пользователя с сайтом.
Обратите внимание: Название
XMLHttpRequest
не означает, что вы можете передавать только XML документы."XML" в названии остался из времён, когда основным форматом для обмена информацией был XML.
Обработка запросов
КонструкторомXMLHttpRequest()
определены несколькотипов ответа. В ответе содержится важная информация о статусе запроса. При указании некоторых типов ответа могут потребоваться дополнительные действия для обработки и получения результата, рассмотрим эти случаи.
Обработка и получение результата из responseXML
Если с помощьюXMLHttpRequest
загрузить XML-документ, в свойствеresponseXML
будет DOM-объект, содержащий распарсенный XML-документ, работать напрямую с которым будет сложно. Есть четыре основных способа анализа этого документа:
- ИспользоватьXPath для обращения (или указания на) к части XML-документа.
- Вручнуюконвертировать XML в строку или объект.
- Использовать
XMLSerializer
для сериализацииDOM-дерева в строку. - Использовать
RegExp
, если вам заранее известна структура документа. Возможно, потребуется удалить переносы строк из документа или учитывать их вRegExp
. Однако, этот способ стоит использовать только в крайнем случае, ведь если XML-документ изменится хотя бы чуть-чуть, то регулярное выражение, скорее всего, уже не подойдёт.
Обратите внимание: Теперь с помощью
responseXML
можно парсить HTML. Подробнее читайте в статьеHTML в XMLHttpRequest.
Получение HTML из responseText
Если вы используетеXMLHttpRequest
для получения содержимого HTML-страницы, в свойствеresponseText
будет "сырой" HTML, работать с которым неудобно. Есть три способа упростить работу с этим "сырым" HTML:
- Использовать свойство
XMLHttpRequest.responseXML
, как показано в статьеHTML в XMLHttpRequest. - Вставить содержимое вфрагмент с помощью
fragment.body.innerHTML
и работать уже с содержимым фрагмента как с DOM-деревом. - Использовать
RegExp
, если вам заранее известна структура HTML. Возможно, потребуется удалить переносы строк из содержимого или учитывать их вRegExp
. Однако, этот способ стоит использовать только в крайнем случае, ведь если HTML изменится хотя бы чуть-чуть, то регулярное выражение, скорее всего, уже не подойдёт.
Работа с двоичными данными
Хотя обычноXMLHttpRequest
используется для отправки и получения текстового содержимого, с его помощью можно обмениваться и двоичными данными. Есть несколько проверенных способов заставитьXMLHttpRequest
посылать двоичные данные. Они заключаются в использовании методаoverrideMimeType()
.
const req = new XMLHttpRequest();req.open("GET", url);// просим извлечь данные в виде двоичной строки без обработкиreq.overrideMimeType("text/plain; charset=x-user-defined");/* ... */
Однако, существуют и более современные способы, так как атрибутresponseType
теперь поддерживает ряд дополнительных типов содержимого, что существенно упрощает отправку и получение двоичных данных.
Для примера рассмотрим фрагмент, где используется "arraybuffer
" как значениеresponseType
для загрузки содержимого как объектаArrayBuffer
, в котором хранятся сырые двоичные данные.
const req = new XMLHttpRequest();req.onload = (e) => { const arraybuffer = req.response; // именно response, не responseText /* ... */};req.open("GET", url);req.responseType = "arraybuffer";req.send();
Больше примеров в статьеОтправка и получение бинарных данных.
Отслеживание прогресса загрузки
XMLHttpRequest
позволяет подписываться на различные события, которые могут произойти в процессе обработки запроса: периодические уведомления о состоянии запроса, сообщения об ошибках и так далее.
СледуяспецификацииXMLHttpRequest
поддерживает событиеprogress
и реализует интерфейсProgressEvent
. Для получения информации о прогрессе загрузки используйте события:
progress
Наступает каждый раз при изменении объёма переданных данных.
load
Наступает по завершению передачи, когда все данные доступны в
response
.
const req = new XMLHttpRequest();req.addEventListener("progress", updateProgress);req.addEventListener("load", transferComplete);req.addEventListener("error", transferFailed);req.addEventListener("abort", transferCanceled);req.open();// ...// отслеживание прогресса передачи от сервера к клиенту (загрузка)function updateProgress(event) { if (event.lengthComputable) { const percentComplete = (event.loaded / event.total) * 100; // ... } else { // невозможно вычислить состояние загрузки, так как размер неизвестен }}function transferComplete(evt) { console.log("Загрузка завершена.");}function transferFailed(evt) { console.log("При загрузке произошла ошибка.");}function transferCanceled(evt) { console.log("Пользователь отменил загрузку.");}
В 3-6 строках добавляются обработчики для различных событий, происходящих при передаче данных с помощьюXMLHttpRequest
.
Важно: Обработчики нужно добавлять до вызова метода
open()
. В противном случаеprogress
-события не будут обработаны.
Обработчик событияprogress
, представленный функциейupdateProgress()
в этом примере, получает количество байт, которое должно быть передано, и количество уже переданных байт в поляхtotal
иloaded.
Но если длина сообщения неизвестна, полеlengthComputable
будет равноfalse
.
События о ходе выполнения есть как у входящих, так и у исходящих передач. Обработчики событий входящих передач задаются для объектаXMLHttpRequest
, как в примере выше, а для исходящих — наXMLHttpRequest.upload
:
const req = new XMLHttpRequest();req.upload.addEventListener("progress", updateProgress);req.upload.addEventListener("load", transferComplete);req.upload.addEventListener("error", transferFailed);req.upload.addEventListener("abort", transferCanceled);req.open();
Обратите внимание: отслеживание прогресса недоступно для протокола
file:
.
События о ходе выполнения наступают для каждого полученного пакета данных, включая последний, поэтому в случае, когда последний пакет получен и соединение закрыто, событиеprogress
всё равно наступит. Это позволяет нам отслеживать прогресс, добавляя обработчик только дляprogress
-события.
Также можно обработать все три события, завершающие загрузку (abort
,load
, orerror
) через событиеloadend
:
req.addEventListener("loadend", loadEnd);function loadEnd(e) { console.log("Передача данных завершена (но мы не знаем, успешно ли).");}
Заметьте, что событиеloadend
никак не сообщает, что вызвало конец передачи. Впрочем, это никак не мешает использовать его, если нужно сделать что-то вне зависимости от причины.
Отправка форм и загрузка файлов
Есть два способа передать данные форм с помощьюXMLHttpRequest
:
- без помощи других API, используя только XHR;
- с помощью
FormData
API.
FormData API – самый простой и быстрый способ, но данные, полученные с его помощью, нельзя превратить в строку с помощьюJSON.stringify. Использование только XHR сложнее, но этот способ самый гибкий и мощный.
Используя толькоXMLHttpRequest
Отправка формы без FormData API в большинстве случаев не требует других API. Единственное исключение,если вам нужно отправить один или несколько файлов, тогда придётся использоватьFileReader
API.
Краткое введение в методы отправки
HTML-форму<form>
можно отправить четырьмя способами:
- использовать метод
POST
и установить атрибутenctype
в значенииapplication/x-www-form-urlencoded
(способ по умолчанию); - использовать метод
POST
и установить атрибутenctype
в значенииtext/plain
; - использовать метод
POST
и установить атрибутenctype
в значенииmultipart/form-data
; - использовать метод
GET
(в этом случае атрибутenctype
будет проигнорирован).
Рассмотрим отправку формы с двумя полями:foo
иbaz
. Если использовать методPOST
, сервер получит строку, похожую на одну из показанных ниже, в зависимости от типа кодирования, который вы используете:
Метод:
POST
; тип кодирования:application/x-www-form-urlencoded
(по умолчанию):Content-Type: application/x-www-form-urlencodedfoo=bar&baz=The+first+line.%0D%0AThe+second+line.%0D%0A
Метод:
POST
; тип кодирования:text/plain
:Content-Type: text/plainfoo=barbaz=The first line.The second line.
Метод:
POST
; тип кодирования:multipart/form-data
:Content-Type: multipart/form-data; boundary=---------------------------314911788813839-----------------------------314911788813839Content-Disposition: form-data; name="foo"bar-----------------------------314911788813839Content-Disposition: form-data; name="baz"The first line.The second line.-----------------------------314911788813839--
А если вы решите использовать методGET
, к адресу формы будет добавлена строка вида:
?foo=bar&baz=The%20first%20line.%0AThe%20second%20line.
Небольшой классический фреймворк
Всё это возможно благодаря браузеру и тегу<form>
. Но если вам требуется выполнить все операции только с помощью JavaScript, вам придётся проинструктировать интерпретатор обовсех выполняемых операциях. Отправка формы с помощьючистого XHR слишком сложна, чтобы рассказать вам о ней во всех деталях. Поэтому мы решили опубликовать здесьцелый (пусть и учебный) фреймворк, который поддерживает все четыре способа отправки и дажезагрузку файлов:
<!doctype html><html lang="en-US"> <head> <meta charset="utf-8" /> <meta name="viewport" content="width=device-width" /> <title>Sending forms with pure AJAX – MDN</title> <script> "use strict"; // :: XHR Form Submit Framework :: // // https://developer.mozilla.org/ru/docs/DOM/XMLHttpRequest/Using_XMLHttpRequest // // This framework is released under the GNU Public License, version 3 or later. // https://www.gnu.org/licenses/gpl-3.0-standalone.html // // Syntax: // // XHRSubmit(HTMLFormElement); const XHRSubmit = (function () { function xhrSuccess() { console.log(this.responseText); // you can get the serialized data through the "submittedData" custom property: // console.log(JSON.stringify(this.submittedData)); } function submitData(data) { const req = new XMLHttpRequest(); req.submittedData = data; req.onload = xhrSuccess; if (data.technique === 0) { // method is GET req.open( "get", data.receiver.replace( /(?:\?.*)?$/, data.segments.length > 0 ? `?${data.segments.join("&")}` : "", ), true, ); req.send(null); } else { // method is POST req.open("post", data.receiver, true); if (data.technique === 3) { // enctype is multipart/form-data const boundary = "---------------------------" + Date.now().toString(16); req.setRequestHeader( "Content-Type", `multipart\/form-data; boundary=${boundary}`, ); req.sendAsBinary( `--${boundary}\r\n` + data.segments.join(`--${boundary}\r\n`) + `--${boundary}--\r\n`, ); } else { // enctype is application/x-www-form-urlencoded or text/plain req.setRequestHeader("Content-Type", data.contentType); req.send(data.segments.join(data.technique === 2 ? "\r\n" : "&")); } } } function processStatus(data) { if (data.status > 0) { return; } // the form is now totally serialized! do something before sending it to the server… // doSomething(data); // console.log("XHRSubmit - The form is now serialized. Submitting..."); submitData(data); } function pushSegment(segment) { this.owner.segments[this.segmentIdx] += segment.target.result + "\r\n"; this.owner.status--; processStatus(this.owner); } function plainEscape(text) { // How should I treat a text/plain form encoding? // What characters are not allowed? this is what I suppose…: // "4\3\7 - Einstein said E=mc2" ----> "4\\3\\7\ -\ Einstein\ said\ E\=mc2" return text.replace(/[\s\=\\]/g, "\\$&"); } function SubmitRequest(target) { const isPost = target.method.toLowerCase() === "post"; this.contentType = isPost && target.enctype ? target.enctype : "application\/x-www-form-urlencoded"; this.technique = isPost ? this.contentType === "multipart\/form-data" ? 3 : this.contentType === "text\/plain" ? 2 : 1 : 0; this.receiver = target.action; this.status = 0; this.segments = []; const filter = this.technique === 2 ? plainEscape : escape; for (const field of target.elements) { if (!field.hasAttribute("name")) { continue; } const fieldType = field.nodeName.toUpperCase() === "INPUT" && field.hasAttribute("type") ? field.getAttribute("type").toUpperCase() : "TEXT"; if (fieldType === "FILE" && field.files.length > 0) { if (this.technique === 3) { // enctype is multipart/form-data for (const file of field.files) { const segmReq = new FileReader(); // Custom properties: segmReq.segmentIdx = this.segments.length; segmReq.owner = this; segmReq.onload = pushSegment; this.segments.push( 'Content-Disposition: form-data; name="' + field.name + '"; filename="' + file.name + '"\r\nContent-Type: ' + file.type + "\r\n\r\n", ); this.status++; segmReq.readAsBinaryString(file); } } else { // enctype is application/x-www-form-urlencoded or text/plain or // method is GET: files will not be sent! for (const file of field.files) { this.segments.push( `${filter(field.name)}=${filter(file.name)}`, ); } } } else if ( (fieldType !== "RADIO" && fieldType !== "CHECKBOX") || field.checked ) { // NOTE: this will submit _all_ submit buttons. Detecting the correct one is non-trivial. // field type is not FILE or is FILE but is empty. if (this.technique === 3) { // enctype is multipart/form-data this.segments.push( `Content-Disposition: form-data; name="${field.name}"\r\n\r\n${field.value}\r\n`, ); } else { // enctype is application/x-www-form-urlencoded or text/plain or method is GET this.segments.push( `${filter(field.name)}=${filter(field.value)}`, ); } } } processStatus(this); } return (formElement) => { if (!formeElement.action) { return; } new SubmitRequest(formElement); }; })(); </script> </head> <body> <h1>Sending forms with XHR</h1> <h2>Using the GET method</h2> <form action="register.php" method="get" onsubmit="AJAXSubmit(this); return false;"> <fieldset> <legend>Registration example</legend> <p> <label>First name: <input type="text" name="firstname" /></label ><br /> <label>Last name: <input type="text" name="lastname" /></label> </p> <p> <input type="submit" value="Submit" /> </p> </fieldset> </form> <h2>Using the POST method</h2> <h3>Enctype: application/x-www-form-urlencoded (default)</h3> <form action="register.php" method="post" onsubmit="AJAXSubmit(this); return false;"> <fieldset> <legend>Registration example</legend> <p> <label>First name: <input type="text" name="firstname" /></label> <br /> <label>Last name: <input type="text" name="lastname" /></label> </p> <p> <input type="submit" value="Submit" /> </p> </fieldset> </form> <h3>Enctype: text/plain</h3> <form action="register.php" method="post" enctype="text/plain" onsubmit="AJAXSubmit(this); return false;"> <fieldset> <legend>Registration example</legend> <p> <label >Your name: <input type="text" name="user" /> </label> </p> <p> <label >Your message:<br /> <textarea name="message" cols="40" rows="8"></textarea> </label> </p> <p> <input type="submit" value="Submit" /> </p> </fieldset> </form> <h3>Enctype: multipart/form-data</h3> <form action="register.php" method="post" enctype="multipart/form-data" onsubmit="AJAXSubmit(this); return false;"> <fieldset> <legend>Upload example</legend> <p> <label>First name: <input type="text" name="firstname" /></label ><br /> <label>Last name: <input type="text" name="lastname" /></label><br /> Sex: <input type="radio" name="sex" value="male" /> <label for="sex_male">Male</label> <input type="radio" name="sex" value="female" /> <label for="sex_female">Female</label><br /> Password: <input type="password" name="secret" /><br /> <label >What do you prefer: <select name="image_type"> <option>Books</option> <option>Cinema</option> <option>TV</option> </select> </label> </p> <p> <label >Post your photos: <input type="file" multiple name="photos[]" /> </label> </p> <p> <input type="checkbox" name="vehicle[]" value="Bike" /> <label for="vehicle_bike">I have a bike</label><br /> <input type="checkbox" name="vehicle[]" value="Car" /> <label for="vehicle_car">I have a car</label> </p> <p> <label >Describe yourself:<br /> <textarea name="description" cols="50" rows="8"></textarea> </label> </p> <p> <input type="submit" value="Submit" /> </p> </fieldset> </form> </body></html>
Чтобы воспользоваться фреймворком, создайте страницу с названиемregister.php (и укажите его в атрибутеaction
одной из форм в примере) сминимальным содержимым:
<?php /* register.php */ header("Content-type: text/plain"); /* NOTE: You should never use `print_r()` in production scripts, or otherwise output client-submitted data without sanitizing it first. Failing to sanitize can lead to cross-site scripting vulnerabilities. */ echo ":: data received via GET ::\n\n"; print_r($_GET); echo "\n\n:: Data received via POST ::\n\n"; print_r($_POST); echo "\n\n:: Data received as \"raw\" (text/plain encoding) ::\n\n"; if (isset($HTTP_RAW_POST_DATA)) { echo $HTTP_RAW_POST_DATA; } echo "\n\n:: Files received ::\n\n"; print_r($_FILES);>
Для активации выполните код:
XHRSubmit(myForm);
Важно: Наш фреймворк использует
FileReader
API для передачи файлов. Это новый API и его невозможно использовать в IE9 и ниже. В связи с этим, загрузки только с использованием AJAX воспринимаются лишь какэкспериментальные. Если вам не требуется загружать бинарные файлы, то данный фреймворк работает в большинстве современных браузеров.
Обратите внимание: Лучший способ отправить бинарные данные – использовать
ArrayBuffers
илиBlobs
в связке с методомsend()
и методомreadAsArrayBuffer()
изFileReader
API. Но так как цель нашего примера – поддержка возможности представить сырые данныев виде строки, мы использовали методsendAsBinary()
в связке сreadAsBinaryString()
изFileReader
API. Таким образом, приведенный выше код имеет смысл использовать только в том случае, если вы имеете дело с небольшими файлами. Если вы не планируете загружать двоичное содержимое, вместо этого воспользуйтесьFormData
API.
Используя FormData
TheFormData
constructor lets you compile aset of key/value pairs to send usingXMLHttpRequest
. Its primary use is insending form data, but can also be used independently from a form in order to transmituser keyed data. The transmitted data is in the same format the form'ssubmit()
method uses to send data, if the form's encoding type were set to"multipart/form-data". FormData objects can be utilized in a number of ways with anXMLHttpRequest
. For examples, and explanations of how one can utilizeFormData with XMLHttpRequests, see theUsing FormData Objects page. For didactic purposes here isatranslation ofthe previous example transformed to use theFormData
API. Note the brevity of the code:
<!doctype html><html lang="en-US"> <head> <meta charset="UTF-8" /> <meta name="viewport" content="width=device-width" /> <title>Sending forms with FormData – MDN</title> <script> "use strict"; function xhrSuccess () { console.log(this.responseText); } function XHRSubmit (formElement) { if (!formElement.action) { return; } const req = new XMLHttpRequest(); req.onload = xhrSuccess; if (fFormElement.method.toLowerCase() === "post") { req.open("post", formElement.action); req.send(new FormData(formElement)); } else { let search = ""; for (const field of formElement.elements) { if (!field.hasAttribute("name")) { continue; } const fieldType = field.nodeName.toUpperCase() === "INPUT" && oField.hasAttribute("type") ? field.getAttribute("type").toUpperCase() : "TEXT"; if (fieldType === "FILE") { for (const file of field.files) { search += `&${escape(field.name)}=${escape(file.name)}`; } else if ((fieldType !== "RADIO" && fieldType !== "CHECKBOX") || field.checked) { search += `&${escape(field.name)}=${escape(field.value)}`; } } req.open("get", formElement.action.replace(/(?:\?.*)?$/, search.replace(/^&/, "?")), true); req.send(null); } } </script> </head> <body> <h1>Sending forms with FormData</h1> <h2>Using the GET method</h2> <form action="register.php" method="get" onsubmit="AJAXSubmit(this); return false;"> <fieldset> <legend>Registration example</legend> <p> First name: <input type="text" name="firstname" /><br /> Last name: <input type="text" name="lastname" /> </p> <p> <input type="submit" value="Submit" /> </p> </fieldset> </form> <h2>Using the POST method</h2> <h3>Enctype: application/x-www-form-urlencoded (default)</h3> <form action="register.php" method="post" onsubmit="AJAXSubmit(this); return false;"> <fieldset> <legend>Registration example</legend> <p> First name: <input type="text" name="firstname" /><br /> Last name: <input type="text" name="lastname" /> </p> <p> <input type="submit" value="Submit" /> </p> </fieldset> </form> <h3>Enctype: text/plain</h3> <p>The text/plain encoding is not supported by the FormData API.</p> <h3>Enctype: multipart/form-data</h3> <form action="register.php" method="post" enctype="multipart/form-data" onsubmit="AJAXSubmit(this); return false;"> <fieldset> <legend>Upload example</legend> <p> First name: <input type="text" name="firstname" /><br /> Last name: <input type="text" name="lastname" /><br /> Sex: <input type="radio" name="sex" value="male" /> <label for="sex_male">Male</label> <input type="radio" name="sex" value="female" /> <label for="sex_female">Female</label><br /> Password: <input type="password" name="secret" /><br /> What do you prefer: <select name="image_type"> <option>Books</option> <option>Cinema</option> <option>TV</option> </select> </p> <p> Post your photos: <input type="file" multiple name="photos[]" /> </p> <p> <input type="checkbox" name="vehicle[]" value="Bike" /> <label for="vehicle_bike">I have a bike</label><br /> <input type="checkbox" name="vehicle[]" value="Car" /> <label for="vehicle_car">I have a car</label> </p> <p> Describe yourself:<br /> <textarea name="description" cols="50" rows="8"></textarea> </p> <p> <input type="submit" value="Submit" /> </p> </fieldset> </form> </body></html>
Примечание:As we said,FormData
objects are notstringifiable objects. If you want to stringify a submitted data, usethe previouspure-AJAX example. Note also that, although in this example there are somefile
<input>
fields,when you submit a form through theFormData
API you do not need to use theFileReader
API also: files are automatically loaded and uploaded.
Получаем дату последнего изменения
function getHeaderTime() { console.log(this.getResponseHeader("Last-Modified")); // Дата вида GMTString или null}const req = new XMLHttpRequest();req.open( "HEAD", // используется HEAD только если сервер требует заголовки "yourpage.html",);req.onload = getHeaderTime;req.send();
Do something when last modified date changes
Let's create two functions:
function getHeaderTime() { const lastVisit = parseFloat( window.localStorage.getItem(`lm_${this.filepath}`), ); const lastModified = Date.parse(this.getResponseHeader("Last-Modified")); if (isNaN(lastVisit) || lastModified > lastVisit) { window.localStorage.setItem(`lm_${this.filepath}`, Date.now()); isFinite(lastVisit) && this.callback(lastModified, lastVisit); }}function ifHasChanged(URL, callback) { const req = new XMLHttpRequest(); req.open("HEAD" /* use HEAD - we only need the headers! */, URL); req.callback = callback; req.filepath = URL; req.onload = getHeaderTime; req.send();}
And to test:
// Let's test the file "yourpage.html"ifHasChanged("yourpage.html", function (modified, visit) { console.log( `The page '${this.filepath}' has been changed on ${new Date( nModified, ).toLocaleString()}!`, );});
If you want to know if the current page has changed, refer to the article aboutdocument.lastModified
.
Межсайтовые XMLHttpRequest
Современные браузеры поддерживают межсайтовые запросы по стандартуCross-Origin Resource Sharing (CORS). Для этого серверу необходимо дополнительно указывать заголовокorigin
. В противном случае, выбрасывается исключениеINVALID_ACCESS_ERR
.
Обход кеширования
Для межсайтового обхода кеширования в конец URL-запроса достаточно добавить случайную строку в GET-параметры, то есть сразу после «?», например:
http://foo.com/bar.html -> http://foo.com/bar.html?12345http://foo.com/bar.html?foobar=baz -> http://foo.com/bar.html?foobar=baz&12345
Таким образом, каждый новый запрос будет происходить по новому URL, не браться из кэша.
Автоматизировать этот подход можно следующим образом:
const req = new XMLHttpRequest();req.open("GET", url + (/\?/.test(url) ? "&" : "?") + new Date().getTime());req.send(null);
Безопасность
Рекомендуемый способ разрешить межсайтовые запросы - использовать HTTP-заголовокAccess-Control-Allow-Origin
в ответе на XMLHttpRequest.
XMLHttpRequests был остановлен
Если в завершение XMLHttpRequest вы получаетеstatus=0
иstatusText=null
– это означает, что запрос не был разрешен к выполнению. Его статус осталсяUNSENT
. Частая причина, что указанныйXMLHttpRequest
origin (во время создания XMLHttpRequest) был изменён в следствии вызоваopen()
. Такое может произойти, например, когда есть XMLHttpRequest, который запускается при событии onunload окна. XMLHttpRequest создается, когда окно, которое должно быть закрыто, всё ещё существует, но отправка запроса (другими словами, вызовopen()
) происходит, когда это окно уже потеряло свой фокус, а другое – получило. Наиболее эффективный способ избежать этой проблемы - установить слушателя на событие нового окнаDOMActivate
, которое устанавливается, как только у закрытого окна срабатывает событиеunload
.
Спецификации
Specification |
---|
XMLHttpRequest # interface-xmlhttprequest |