ネイティブ JS で非同期リクエストを使用して Ajax を実装する方法

php中世界最好的语言
リリース: 2018-04-16 10:48:26
オリジナル
2071 人が閲覧しました

今回は、ネイティブ JS 非同期リクエストを使用して Ajax を実装する方法を説明します。Ajax を実装するためのネイティブ JS 非同期リクエストの 注意事項 については、次のとおりです。

フロントエンド ページの開発プロセスでは、フォーム データの非同期送信やページの非同期更新などに、Ajax リクエストがよく使用されます。

一般的に、Jquery で

を使用すると非常に便利ですが、ajax 機能が必要だからといって Jquery を導入するのは費用対効果が低い場合があります。 $.ajax,$.post,$.getJSON

次に、ネイティブ JavaScrpit を使用して単純な Ajax リクエストを実装し、Ajax リクエストにおけるクロスドメイン アクセスの問題と、複数の Ajax リクエストのデータ同期の問題について説明します。

JavaScriptはAjax非同期リクエストを実装します

シンプルな Ajax リクエストの実装

Ajax リクエストの原理は、XMLHttpRequest オブジェクトを作成し、このオブジェクトを使用してリクエストを非同期に送信することです。具体的な実装については、次のコードを参照してください:

function ajax(option) {
  // 创建一个 XMLHttpRequest 对象
  var xhr = window.XMLHttpRequest ? new XMLHttpRequest() : new ActiveXObject("Microsoft.XMLHTTP"),
    requestData = option.data,
    requestUrl = option.url,
    requestMethod = option.method;
  // 如果是GET请求,需要将option中的参数拼接到URL后面
  if ('POST' != requestMethod && requestData) {
    var query_string = '';
    // 遍历option.data对象,构建GET查询参数
    for(var item in requestData) {
      query_string += item + '=' + requestData[item] + '&';
    }
    // 注意这儿拼接的时候,需要判断是否已经有 ?
    requestUrl.indexOf('?') > -1
      ? requestUrl = requestUrl + '&' + query_string
      : requestUrl = requestUrl + '?' + query_string;
    // GET 请求参数放在URL中,将requestData置为空
    requestData = null;
  }
  // ajax 请求成功之后的回调函数
  xhr.onreadystatechange = function () {
    // readyState=4表示接受响应完毕
    if (xhr.readyState == ("number" == typeof XMLHttpRequest.DONE ? XMLHttpRequest.DONE : 4)) {
      if (200 == xhr.status) { // 判断状态码
        var response = xhr.response || xhr.responseText || {}; // 获取返回值
        // if define success callback, call it, if response is string, convert it to json objcet
        console.log(response);
        option.success && option.success(response); // 调用成功的回调函数处理返回值
        // 可以判断返回数据类型,对数据进行JSON解析或者XML解析
        // option.success && option.success('string' == typeof response ? JSON.parse(response) : response);
      } else {
        // if define error callback, call it
        option.error && option.error(xhr, xhr.statusText);
      }
    }
  };
  // 发送ajax请求
  xhr.open(requestMethod, requestUrl, true);
  // 请求超时的回调
  xhr.ontimeout = function () {
    option.timeout && option.timeout(xhr, xhr.statusText);
  };
  // 定义超时时间
  xhr.timeout = option.timeout || 0;
  // 设置响应头部,这儿默认设置为json格式,可以定义为其他格式,修改头部即可
  xhr.setRequestHeader && xhr.setRequestHeader('Content-Type', 'application/json;charset=utf-8');
  xhr.withCredentials = (option.xhrFields || {}).withCredentials;
  // 这儿主要用于发送POST请求的数据
  xhr.send(requestData);
}
ログイン後にコピー
上記のコードには詳細なコメントがあります。ajax の原理は一般に、XMLHttpRequest オブジェクトを使用してデータを送信します。このオブジェクトについて補足説明します。

多くのブール式の短絡プロパティは、条件ステートメントの記述を置き換えるためにコード内で使用されます。ブール式の短絡プロパティをうまく利用すると、多数の単純な条件ステートメントを簡素化できます。 ^_^

XMLHttpRequest オブジェクトの基本プロパティ

ReadyState 属性には 5 つのステータス値があります:

0: 未初期化: 初期化されていません。 XMLHttpRequest オブジェクトは作成されましたが、初期化されていません。

1: はい読み込み中: 送信する準備ができています。

2: はロードされました、: は送信されましたが、応答はまだ受信されていません。

3: 対話型です。応答は受信されていますが、まだ受信されていません。

4: はい完了: 応答は受け入れられます。

responseText: サーバーから返された応答テキスト。これは、readyState>=3 の場合にのみ値を持ちます。 readyState=3 の場合、返される応答テキストは不完全です。readyState=4 の場合のみ、完全な応答テキストが受信されます。 responseXML: 応答情報は XML であり、Dom オブジェクトに解析できます。 status: サーバーの

HTTP ステータス コード が 200 の場合は OK を意味し、404 は見つからないことを意味します。 statusText: サーバーの http ステータス コードのテキスト。たとえば、OK、見つかりません。

XMLHttpRequestオブジェクトの基本メソッド

open(method, url, asyn): XMLHttpRequest オブジェクトを開きます。メソッドには、get、post、delete、および put が含まれます。 url は、要求されたリソースのアドレスです。 3 番目のパラメータは、非同期を使用するかどうかを示します。 Ajaxの特徴は非同期送信であるため、デフォルトはtrueです。同期が使用される場合は False。 send(body): Ajax リクエストを送信します。送信する内容は必須パラメータにすることができます。パラメータがない場合は、(null) を直接送信します

使い方

上記で定義した ajax 関数を呼び出し、対応するオプションとパラメーターを渡すだけです。

りー

クロスドメインリクエストの問題

Ajax リクエストを使用する場合、クロスドメインリクエストという 1 つの問題に注意する必要があります。 特別な手段を使用しない場合、クロスドメイン リクエスト: 他のドメイン名およびポートで URL リソースをリクエストすると、エラー メッセージが報告されます。 Access-Control-Allow-Origin 関連のエラー。主な理由は、クロスドメイン リソース要求を行うことができないと規定するブラウザーの同一オリジン ポリシー制限です。

解決策

ここでは、いくつかの解決策を簡単に説明します。 Ajax ヘッダーにクロスドメイン リクエストを許可するヘッダーを追加する この方法では、サーバーがクロスドメイン リクエストを許可するヘッダーの追加に協力する必要もあります。以下は、Allow Cross-domain ヘッダーを POST リクエストに追加する PHP の例です:

ajax({
  url: '/post.php',
  data: {
    name: 'uusama',
    desc: 'smart'
  },
  method: 'GET',
  success: function(ret) {
    console.log(ret);
  }
});
ログイン後にコピー
動的 scrpit タグを使用して、動的に scrpit タグを作成し、要求されたアドレスを指すメソッド (つまり、JSONP メソッド) では、URL の後にコールバック関数を結合する必要があります。コールバック関数は、タグが正常に読み込まれた後に呼び出されます。

りー

コールバック関数はグローバル関数として設定する必要があります:

// 指定允许其他域名访问 
header('Access-Control-Allow-Origin:*'); 
// 响应类型 
header('Access-Control-Allow-Methods:POST'); 
// 响应头设置 
header('Access-Control-Allow-Headers:x-requested-with,content-type');
ログイン後にコピー

複数の Ajax リクエストによるデータ同期の問題

単一の Ajax 戻りデータの非同期処理

多个ajax请求互不相关,它们在被调用以后发送各自请求,请求成功以后调用自己的回调方法,互不影响。 因为ajax请求异步的特性,所有一些依赖于请求完成之后的操作我们都需要放在回调函数内部,否则的话,你在回调函数外面读取到的值是空。看下面的例子:

var result = null;
ajax({
  url: '/get.php?id=1',
  method: 'GET',
  success: function(ret) {
    result = ret;
  }
});
console.log(result); // 输出 null
ログイン後にコピー

虽然我们在回调函数里面设置了result的值,但是在最后一行 console.log(result); 输出为空。 因为ajax请求是异步的,程序执行到最后一行的时候,请求并没有完成,值并没有来得及修改。 这儿我们应该把 console.log(result) 相关的处理,放在 success 回调函数中才可以。

多个ajax返回数据问题

如果有多个ajax请求,情况会变得有些复杂。 如果多个ajax请求是按照顺序执行的,其中一个完成之后,才能进行下一个,则可以把后面一个请求放在前一后请求的回调中。 比如有两个ajax请求,其中一个请求的数据依赖于另外一个,则可以在第一个请求的回调里面再进行ajax请求:

// 首先请求第一个ajax
ajax({
  url: '/get1.php?id=1',
  success: function(ret1) {
    // 第一个请求成功回调以后,再请求第二个
    if (ret1) {
      ajax({
        url: '/get2.php?id=4',
        success:function(ret2) {
          console.log(ret1);
          console.log(ret2)
        }
      })
    }
  }
});
// 也可以写成下面的形式
var ajax2 = function(ret1) {
  ajax({
    url: '/get2.php?id=4',
    success:function(ret2) {
      console.log(ret1);
      console.log(ret2)
    }
  });
};
ajax({
  url: '/get1.php?id=1',
  success: function(ret1) {
    if(ret1){
      ajax2(ret1);
    }
  }
});
ログイン後にコピー

如果不关心不同的ajax请求的顺序,而只是关心所有请求都完成,才能进行下一步。 一种方法是可以在每个请求完成以后都调用同一个回调函数,只有次数减少到0才执行下一步。

var count = 3, all_ret = []; // 调用3次
ajax({
  url: '/get1.php?id=1',
  success:function(ret) {
    callback(ret);
  }
});
ajax({
  url: '/get2.php?id=1',
  success:function(ret) {
    callback(ret);
  }
});
ajax({
  url: '/get3.php?id=1',
  success:function(ret) {
    callback(ret);
  }
});
function callback(ret) {
  if (count > 0) {
    count--;
    // 可以在这儿保存 ret 到全局变量
    all_ret.push(ret);
    return;
  } else { // 调用三次以后
    // todo
    console.log(ret);
  }
}
ログイン後にコピー

另一种方法是设置一个定时任务去轮训是否所有ajax请求都完成,需要在每个ajax的成功回调中去设置一个标志。 这儿可以用是否获得值来判断,也可以设置标签来判断,用值来判断时,要注意设置的值和初始相同的情况。

var all_ret = {
  ret1: null, // 第一个ajax请求标识
  ret2: null, // 第二个ajax请求标识
  ret3: null, // 第三个ajax请求标识
};
ajax({
  url: '/get1.php?id=1',
  success:function(ret) {
    all_ret['ret1'] = ret; // 设置第一个ajax请求完成,把结果更新
  }
});
ajax({
  url: '/get2.php?id=1',
  success:function(ret) {
    all_ret['ret2'] = ret; // 设置第二个ajax请求完成,把结果更新
  }
});
ajax({
  url: '/get3.php?id=1',
  success:function(ret) {
    all_ret['ret3'] = ret; // 设置第三个ajax请求完成,把结果更新
  }
});
var repeat = setInterval(function(){
  // 循环检查是否所有设置的ajax请求结果的值是否都已被更改,都已被更改说明所有ajax请求都已完成
  for(var item in all_ret) {
    if (all_ret[item] === null){
      return;
    }
  }
  // todo, 到这儿所有ajax请求均已完成
  clearInterval(repeat);
}, 50);
ログイン後にコピー

PS:下面看下ajax异步请求实例代码,具体代码如下所示:

<%@ page language="java" contentType="text/html; charset=UTF-8"
  pageEncoding="UTF-8"%>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>获得书籍列表</title>
<script type="text/javascript">
var xmlhttp;
var txt,x,xx,i;
function loadXMLDoc(url,cfunc)
{
if(window.XMLHttpRequest)
{
// code for IE7+, Firefox, Chrome, Opera, Safari
xmlhttp = new XMLHttpRequest();
}
else
{
// code for IE6, IE5
xmlhttp = new ActiveXObject("Microsoft.XMLHTTP");
}
 xmlhttp.onreadystatechange = cfunc;
 xmlhttp.open("GET", "<%=request.getContextPath()%>"+url, true);
 xmlhttp.send();
}
function myFunction1()
{
loadXMLDoc("/xmls/books.xml",function(){
if(xmlhttp.readyState==4 && xmlhttp.status==200)
{
var xmlDoc = xmlhttp.responseXML;
txt = "";
x = xmlDoc.getElementsByTagName_r("title");
for(i=0;i<x.length;i++)
{
txt = txt + x[i].childNodes[0].nodeValue+"<br/>";
}
document.getElementByIdx_x("myp").innerHTML = txt;
}
});
}
  function myFunction2()
  {
  loadXMLDoc("/text/test1.txt",function(){
  if(xmlhttp.readyState==4 && xmlhttp.status==200)
  {
  document.getElementByIdx_x("myp").innerHTML = xmlhttp.responseText;
  }
  });
  }
  function myFunction3()
  {
  loadXMLDoc("/xmls/cd_catalog.xml",function(){
  if(xmlhttp.readyState==4 && xmlhttp.status==200)
  {
  txt="<table border=&#39;1&#39;><tr><th>Title</th><th>Artist</th></tr>"
  x= xmlhttp.responseXML.documentElement.getElementsByTagName_r("CD");
  for(i=0;i<x.length;i++)
  {
  txt = txt + "<tr>";
  xx = x[i].getElementsByTagName_r("TITLE");
  {
  try{
  txt = txt + "<td>" + xx[0].firstChild.nodeValue +"</td>";
   }
  catch(er)
  {
  txt = txt +"<td></td>";
  }
  xx = x[i].getElementsByTagName_r("ARTIST");
  try
  {
  txt = txt + "<td>" + xx[0].firstChild.nodeValue +"</td>";
  }
  catch(er)
  {
  txt = txt + "<td></td>";
  }
  }
  txt = txt + "</tr>"
  }
  txt = txt + "</table>";
  document.getElementByIdx_x("myp").innerHTML =txt;
  }
  });
  }
</script>
</head>
<body>
<h2>My Book Collection:</h2>
<button type="button" onClick="myFunction1()">获得我的图书收藏列表</button>
<button type="button" onClick="myFunction2()">这是不同的请求</button>
<button type="button" onClick="myFunction3()">获取CD信息</button>
<p id="myp"></p>
</body>
</html>
ログイン後にコピー

相信看了本文案例你已经掌握了方法,更多精彩请关注php中文网其它相关文章!

推荐阅读:



以上がネイティブ JS で非同期リクエストを使用して Ajax を実装する方法の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

関連ラベル:
ソース:php.cn
このウェブサイトの声明
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。
最新の問題
人気のチュートリアル
詳細>
最新のダウンロード
詳細>
ウェブエフェクト
公式サイト
サイト素材
フロントエンドテンプレート