ajax无法访问,Ajax不能跨域访问的解决方案

本文详细介绍了跨域访问的概念,通过分析URL的组成部分解释了何为跨域。文章以Ajax为例,说明其不支持跨域访问的问题,并提出利用src属性,特别是img和script标签来实现跨域请求。通过实例展示了如何利用PHP后端接口和JavaScript的回调函数处理跨域数据。最后,演示了如何利用百度搜索提示接口,实现动态加载搜索建议,从而实现跨域访问的效果。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

19fa2da350ff

文章介绍

这是一篇,引导文吧... 因为写这篇文章时,实在想不出该如何分序。因此以实现跨域访问为目的,从基础知识往上写。最后以百度搜索智能提示为例,来讲解跨域的具体应用!

内容

首先,我们得明确什么是跨域,这里先了解一下url中各组成部分

以百度为例:

https://www.baidu.com:80

协议:https://

二级域名:www

一级域名:baidu.com

端口号:80

以上4个有一个不同即为跨域访问,比如你当前页面在https://www.baidu.com:80,你去用Ajax请求https://tieba.baidu.com:80的数据,就相当于跨域访问!

在Ajax中,是不支持跨域访问的,所以www.baidu.com拿不到tieba.baidu.com上的数据。

那么这里就要用到跨域访问的技巧,虽然Ajax不支持,但我们可以利用src这个属性达到目的。

对于src这个属性,相信很多人都会想到img标签!我们都知道img标签的src如果设置为一个网络地址时,那么就会去使用该网络地址的图片资源。

比如src=https://ss0.bdstatic.com/5aV1bjqh_Q23odCf/static/superman/img/logo/bd_logo1_31bdc765.png

那么img标签就会请求该url的数据返回来,其实这就已经是跨域访问了。因为你当前的页面并不在https://ss0.bdstatic.com上,但你却成功访问到了它的资源。

因此我们可以利用src这个利器,达到我们跨域访问的目的。

不过使用src之前,得先了解一下Ajax利用请求回来的响应数据执行回调的一种方法:

//先定义一个函数,等会利用该函数执行回调

function fun (obj) {

console.log(obj);

}

//以下均为ajax请求,粗略看即可

var url = 'test.php';//访问当前目录的php文件

var xhr = new XMLHttpRequest();

xhr.open('get',url);

xhr.send();

xhr.onreadystatechange = function () {

if (xhr.readyState == 4) {

if (xhr.status >=200 && xhr.status <300 || xhr.status == 304) {

//请求成功后,拿到返回字符串并使用eval执行

eval(xhr.responseText);

} else {

console.log('请求失败');

}

}

}

为了拿到请求数据,下面我们利用php简单制作一个后端接口test.php

//创建一个php对象 name:kevin,age:23

$obj = array("name"=>"Kevin","age"=>"23");

//将php对象编码为json格式的字符串

$json = json_encode($obj);

//后端给客户端返回一个字符串 点为字符串拼接(相当于js中加号)

echo("fun(" . $json . ")");

请求成功后eval(xhr.responseText);这一句可以一步步分解为如下代码:

eval(xhr.responseText);

eval(fun({"name":"Kevin", "age":"23"}));

fun({"name":"Kevin", "age":"23"});

console.log({"name":"Kevin", "age":"23"})

因此控制台打印如下:

19fa2da350ff

浏览器控制台

以上我们实现了:根据后端返回的数据来决定具体执行某一个函数,并且利用请求回来的数据传参!

但是以上并没有解决我们跨域需求,因为我请求的后端接口,依然在当前域!

那么我刚才说了,利用src,我们就能实现跨域。假如我们使用img标签,使其src="test.php"。这样我们虽然拿到了数据,但是img标签并不会像eval函数一样帮我们执行数据,这样数据就得不到利用。

所以对于实现跨域,我们有这样的需求:

有src属性

可以像eval函数一样将数据执行!

满足以上条件的确实有,那就是script标签。

平时我们在script标签内写代码,其实写的就是字符串,并且script标签会帮我们执行。

并且我们平时经常会用到script标签的src属性,比如引入框架时,下面以jQuery为例:

其实它就相当于执行了如下:

//一大堆jQ代码字符串...

因此我们可以利用script标签就行跨域访问,这里以百度为例:

这是百度搜索时,搜索提示的数据接口

https://sp0.baidu.com/5a1Fazu8AA54nxGko9WTAnF6hhy/su?wd=关键字&cb=方法名

比如我搜索a,页面如下:

19fa2da350ff

百度搜索提示

那么我利用接口url也会得到上面数据,如图:

19fa2da350ff

接口获取的数据

这里方法名会根据我们的参数返回,因此你请求数据成功后要执行什么函数,那么就往cb传该函数的方法名!

下面我用尽量少的代码实现一下 百度搜索提示

//回调方法,数据返回后触发

function callBack(obj) {

//从上面图知道关键字数组位于数据的s属性中

var array = obj.s;

//每次触发先清空以前数据,再添加

ul.innerHTML = "";

for (var i = 0; i < array.length; i++) {

var li = document.createElement('li');

li.innerText = array[i];

ul.appendChild(li);

}

}

//url1 和 url2 为了以后方便拼接

var url1 = 'https://sp0.baidu.com/5a1Fazu8AA54nxGko9WTAnF6hhy/su?wd=';

var url2 = '&cb=';

var ul = document.getElementById('ul');

var input = document.getElementById('input');

//监听输出框的键盘输入

input.onkeyup = function () {

//创建script标签

var script = document.createElement('script');

//把输入框的值和方法名作为url参数

script.src = url1 + this.value+ url2 + callBack.name;

//把script标签添加到body,那么就会执行代码

document.body.appendChild(script);

};

最后实现效果如下:

19fa2da350ff

效果

代码仅以实现效果为目的,写得很简短,大家可以直接拷贝到自己电脑尝试~

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值