Maison >interface Web >js tutoriel >Explication graphique détaillée des problèmes inter-domaines ajax (avec code)
Cette fois, je vous apporte une explication graphique détaillée des problèmes inter-domaines ajax (avec code). Quelles sont les précautions pour les problèmes inter-domaines ajax ? Voici des cas réels, jetons un coup d'œil.
Interdomaine
Restrictions de la politique d'origine
La politique d'origine empêche les scripts chargés à partir d'un domaine d'être récupérés ou manipulés Document propriétés sur un autre domaine. Autrement dit, le domaine de l'URL demandée doit être le même que le domaine de la page Web actuelle. Cela signifie que le navigateur isole le contenu de différentes sources pour empêcher les opérations entre elles.
Solution
De manière générale, il existe deux manières courantes, l'une du côté serveur et l'autre du point de vue du client. Les deux présentent des avantages et des inconvénients, et la méthode à utiliser nécessite une analyse spécifique.
Le serveur définit l'en-tête de réponse
Le serveur proxy
Le client utilise un script mécanisme de rappel.
Méthode 1
Le mot-clé Access-Control-Allow-Origin
ne prendra effet que s'il est défini côté serveur . Autrement dit, même si vous utilisez
xmlhttprequest.setHeaderREquest('xx','xx');sur le client, cela n'aura aucun effet.
Requête ajax normale
Simulons le cas d'implémentation d'une requête ajax non interdomaine. test1.htmlnbsp;html> <meta> <title>ajax 测试</title> <input> <p></p> <script> var xhr = new XMLHttpRequest(); var url = 'http://localhost/learn/ajax/test1.php'; function crossDomainRequest() { document.getElementById('content').innerHTML = "<font color='red'>loading..."; // 延迟执行 setTimeout(function () { if (xhr) { xhr.open('GEt', url, true); xhr.onreadystatechange = handle_response; xhr.send(null); } else { document.getElementById('content').innerText = "不能创建XMLHttpRequest对象"; } }, 3000); } function handle_response() { var container = document.getElementById('content'); if (xhr.readyState == 4) { if (xhr.status == 200 || xhr.status == 304) { container.innerHTML = xhr.responseText; } else { container.innerText = '不能跨域请求'; } } } </script>Le contenu de test1.PHP dans le même répertoire est le suivant :
<?php echo "It Works."; ?>
Requête inter-domaines
Tout à l'heure, le fichier HTML et le fichier PHP étaient tous deux sous le conteneur Apache, il n'y avait donc pas de situation inter-domaines. Maintenant, placez le fichier HTML sur le bureau. et demandez à nouveau les données PHP. Cela crée une telle "requête inter-domaines".Faites attention aux informations de la barre d'adresse du navigateur
Lors de votre nouvelle visite, vous trouverez le message d'erreur suivant. Dans ce cas, une opération courante consiste à définir Access-Control-Allow-Origin. Format : Access-Control-Allow-Origin : domain.com/xx/yy.*Si vous connaissez le nom de domaine du client ou le chemin fixe de la demande, il est préférable de ne pas utiliser la méthode des caractères génériques pour garantir davantage la sécurité. Si vous n'êtes pas sûr, utilisez simplement le caractère générique *. Lorsque le langage de développement back-end est PHP, vous pouvez le définir comme ceci au début du fichier :header("Access-Control-Allow-Origin: *");S'il s'agit d'une page ASPX, vous devez le définir comme ceci (Java est similaire) :
Response.AddHeader("Access-Control-Allow-Origin", "*");À ce moment-là, visitez à nouveau le chemin tout à l'heure. Cette méthode devrait être considérée comme plus couramment utilisée et largement adoptée à sens unique. Dire qu’être agent, c’est un peu trop écrit, mais en fait, ce n’est qu’un messager. Donnons un petit exemple : Xiao Ming aime une fille nommée Xiao Hong en classe 3, mais est trop gênée pour lui demander son QQ et son identifiant WeChat. Ensuite, j'ai demandé à Xiaolan, une fille de ma classe. Venez vous aider à l'obtenir. Xiaolan équivaut donc à un agent. Aidez Xiao Ming à obtenir les coordonnées de Xiao Hong qui n'ont pas pu être obtenues directement. Donnons un exemple pour illustrer ce problème.
Demande inter-domaines directe
Modifiez simplement l'URL tout de suite et laissez ajax demander directement des données à d'autres sites Web.nbsp;html> <meta> <title>ajax 测试</title> <input> <p></p> <script> var xhr = new XMLHttpRequest(); // var url = 'http://localhost/learn/ajax/test1.php'; var url = 'http://api.qingyunke.com/api.php?key=free&appid=0&msg=%E5%93%92%E5%93%92'; function crossDomainRequest() { document.getElementById('content').innerHTML = "<font color='red'>loading..."; // 延迟执行 setTimeout(function () { if (xhr) { xhr.open('GEt', url, true); xhr.onreadystatechange = handle_response; xhr.send(null); } else { document.getElementById('content').innerText = "不能创建XMLHttpRequest对象"; } }, 3000); } function handle_response() { var container = document.getElementById('content'); if (xhr.readyState == 4) { if (xhr.status == 200 || xhr.status == 304) { container.innerHTML = xhr.responseText; } else { container.innerText = '不能跨域请求'; } } } </script>Les résultats sont les suivants :
Activer le mode proxy
Nous utilisons toujours le HTML page tout à l'heure Propre interface :url = 'http://localhost/learn/ajax/test1.php';Les détails sont les suivants :
nbsp;html> <meta> <title>ajax 测试</title> <input> <p></p> <script> var xhr = new XMLHttpRequest(); var url = 'http://localhost/learn/ajax/test1.php'; // var url = 'http://api.qingyunke.com/api.php?key=free&appid=0&msg=%E5%93%92%E5%93%92'; function crossDomainRequest() { document.getElementById('content').innerHTML = "<font color='red'>loading..."; // 延迟执行 setTimeout(function () { if (xhr) { xhr.open('GEt', url, true); xhr.onreadystatechange = handle_response; xhr.send(null); } else { document.getElementById('content').innerText = "不能创建XMLHttpRequest对象"; } }, 3000); } function handle_response() { var container = document.getElementById('content'); if (xhr.readyState == 4) { if (xhr.status == 200 || xhr.status == 304) { container.innerHTML = xhr.responseText; } else { container.innerText = '不能跨域请求'; } } } </script>
然后对应的test1.php应该帮助我们实现数据请求这个过程,把“小红的联系方式”要到手,并返回给“小明”。
<?php $url = 'http://api.qingyunke.com/api.php?key=free&appid=0&msg=hello%20world.'; $result = file_get_contents($url); echo $result; ?>
下面看下代码执行的结果。
jsonp方式
JSONP(JSON with Padding) 灵感其实源于在HTML页面中script标签内容的加载,对于script的src属性对应的内容,浏览器总是会对其进行加载。于是:
克服该限制更理想方法是在 Web 页面中插入动态脚本元素,该页面源指向其他域中的服务 URL 并且在自身脚本中获取数据。脚本加载时它开始执行。该方法是可行的,因为同源策略不阻止动态脚本插入,并且将脚本看作是从提供 Web 页面的域上加载的。但如果该脚本尝试从另一个域上加载文档,就不会成功。
实现的思路就是:
在服务器端组装出客户端预置好的json数据,通过回调的方式传回给客户端。
原生实现
nbsp;html> <meta> <title>ajax 测试</title> <script></script> <input> <input> <p></p> <script> function jsonpcallback(result) { for(var i in result) { alert(i+":"+result[i]); } } var JSONP = document.createElement("script"); JSONP.type='text/javascript'; JSONP.src='http://localhost/learn/ajax/test1.php?callback=jsonpcallback'; document.getElementsByTagName('head')[0].appendChild(JSONP); </script>
服务器端test1.php内容如下:
<?php $arr = [1,2,3,4,5,6]; $result = json_encode($arr); echo "jsonpcallback(".$result.")"; ?>
需要注意的是最后组装的返回值内容。
来看下最终的代码执行效果。
JQuery方式实现
采用原生的JavaScript需要处理的事情还是蛮多的,下面为了简化操作,决定采用jQuery来代替一下。
nbsp;html> <meta> <title>ajax 测试</title> <script></script> <input> <input> <p></p> <script> function later_action(msg) { var element = $("<p><font color='green'>"+msg+"<br />"); $("#content").append(element); } $("#btn").click(function(){ // alert($("#talk").val()); $.ajax({ url: 'http://localhost/learn/ajax/test1.php', method: 'post', dataType: 'jsonp', data: {"talk": $("#talk").val()}, jsonp: 'callback', success: function(callback){ console.log(callback.content); later_action(callback.content); }, error: function(err){ console.log(JSON.stringify(err)); }, }); }); </script>
相应的,test1.php为了配合客户端聊天的需求,也稍微做了点改变。
<?php $requestparam = isset($_GET['callback'])?$_GET['callback']:'callback'; // 青云志聊天机器人接口: http://api.qingyunke.com/api.php?key=free&appid=0&msg=hello // 接收来自客户端的请求内容 $talk = $_REQUEST['talk']; $result = file_get_contents("http://api.qingyunke.com/api.php?key=free&appid=0&msg=$talk"); // 拼接一些字符串 echo $requestparam . "($result)"; ?>
最后来查看一下跨域的效果吧。
总结
至此,关于简单的ajax跨域问题,就算是解决的差不多了。对我个人而言,对于这三种方式有一点点自己的看法。
服务器设置Access-Control-Allow-Origin的方式适合信用度高的小型应用或者个人应用。
代理模式则比较适合大型应用的处理。但是需要一个统一的规范,这样管理和维护起来都会比较方便。
JSONP方式感觉还是比较鸡肋的(有可能是我经验还不足,没认识到这个方式的优点吧(⊙﹏⊙)b)。自己玩玩知道有这么个东西好了。维护起来实在是优点麻烦。
相信看了本文案例你已经掌握了方法,更多精彩请关注php中文网其它相关文章!
推荐阅读:
Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!