原生AJAX
AJAX全称为Asynchronous JavaScript And XML,就是异步的JS和XML,通过AJAX可以在浏览器中向服务器发送异步请求,最大的优势:无需刷新就可获取数据。
AJAX不是新的编程语言,而是一种将现有的标准组合在一起使用的新方式
XML简介
- XML可扩展标记语言。
- XML被设计用来传输和存储数据。HTML用来呈现数据。
- XML和HTML类似,不同的是HTML中都是预定义标签,而XML中没有预定义标签,全都是自定义标签,用来表示一些数据
比如说有一个学生数据:
用XML表示
<student>
<name>孙悟空</name>
<age>18</age>
<gender>男</gender>
</student>
XML目前已被JSON取代
AJAX的特点
AJAX的优点
- 可以无需刷新页面而与服务器端进行通信
- 允许你根据用户事件来更新部分页面内容
AJAX的缺点
- 没有浏览历史,不能回退
- 存在跨域问题(同源)
- SEO不友好(搜索引擎优化 )
JS动态创建数据,搜索引擎不能搜索到,不可通过爬虫找到关键信息
HTTP协议
HTTP(hypertext transport protocol)协议【超文本传输协议】,协议详细规定了浏览器和万维网服务器之间互相通信的规则。
请求报文
重点是格式与参数
行 POST /s?ie=utf-8 HTTP/1.1
头 Host:atguigu.com
Cookie:name=guigu
Content-type:application/x-www-form-urlencoded
User-Agent:chrome 83
空行
体 username=admain&password=admin如果是GET请求,请求体是空的,如果是POST请求,请求体可以不为空
响应报文
行 HTTP/1.1(协议版本) 200(协议状态码) ok(协议字符串)
头 Content-Type:text/html;charset=utf-8
Content-length:2048
Content-encoding:gzip
空行
体 <html>
<head>
<body>
<h1>尚硅谷</h1>
</body>
</head>
</html>
例如:在百度中搜索CSDN,点击F12键,点击Network刷新页面,之后点击第一个链接
点击之后会出现以下几个标签
Preview作为一个响应预览 ,展示解析之后的界面
Header标签中会展示以下页面
以上就包含了响应头(Response Headers)和请求头(Request Headers)
点击请求头之后会出现请求头的内容,但此时没有请求行,此时点击view source,就会出现请求行的内容
Header标签中的Query String Parameters为查询字符串,对请求行中的url进行解析
响应头同理
响应体在Response里
例如在进行登陆页面时,请求体在Form Data中,里面会包含用户输入的账号和密码等信息
Express基于Node.js平台,快速,开放,极简的Web开发框架
简单的express框架使用
//1.引入express
const express=require('express')
// 2.创建应用对象
const app=express()
// 3.创建路由规则
// request 是对请求报文的封装
// response 是对响应报文的封装
app.get('/',(request,response)=>{
// 设置响应
response.send('HELLO EXPRESS')
})
// 4.监听端口启动服务
app.listen(8000,()=>{
console.log("服务已经启动,8000端口监听中.....")
})
运行此代码时终端效果如下:
8000端口则会显示出响应
以下为向响应器发出的请求头和请求行
响应头和响应行如下:
响应体
案例
需求:点击按钮,发送请求之后,将响应体的内容显示在页面上,页面不刷新
服务准备:
//1.引入express
const express=require('express')
// 2.创建应用对象
const app=express()
// 3.创建路由规则
// request 是对请求报文的封装
// response 是对响应报文的封装
// 当客户端浏览器向服务端发送请求时,请求行的第二段路径是/server的话,就会执行回调函数里面的代码,并且由response来作出响应
app.get('/server',(request,response)=>{
// 设置响应头 设置允许跨域
response.setHeader('Access-Control-Allow-Origin','*')
// 设置响应体
response.send('HELLO AJAX')
})
// 4.监听端口启动服务
app.listen(8000,()=>{
console.log("服务已经启动,8000端口监听中.....")
})
发送新的请求时,需将8000端口释放,否则终端会报错提示8000端口已被占用
故要先释放上一次端口,才可继续发送请求,在node终端中输入ctrl+c释放端口即可,node即为上次发送请求的终端,在此终端中输入ctrl+c即可
html以及js代码:
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>AJAX GET请求</title>
<style>
#result{
width: 200px;
height: 100px;
border: solid 1px #000;
}
</style>
</head>
<body>
<button>点击发送请求</button>
<div id="result"></div>
<script>
// 获取button元素
const btn=document.getElementsByTagName('button')[0]
const result=document.getElementById('result')
// 绑定事件
btn.addEventListener('click',function(){
// console.log('test')
// 1.创建对象
// XHR是对AJAX请求做出一个筛选
const xhr=new XMLHttpRequest()
// 2.初始化 设置请求方法和URL
xhr.open('GET','http://127.0.0.1:8000/server')
// 3.发送
xhr.send()
// 4.事件绑定 处理服务端返回的结果
// on when 当.....时候
// readystate 是对象中的属性,表示状态
// 0表示未初始化 1表示open方法已经调用完毕 2表示send方法已经调用完毕 3表示服务端返回的部分结果 4表示服务端返回的所有结果
// change 改变
// 该事件总共会触发四次 0-1一次 1-2一次 2-3一次 3-4一次
xhr.onreadystatechange=function(){
// 处理服务端返回的结果 当状态为4的时候处理,状态为4已经返回所有结果
// 判断(服务端返回了所有的结果)
if(xhr.readyState===4){
// 判断响应状态码 200 404 401 500
// 2XX 表示成功
if(xhr.status>=200&&xhr.status<300){
// 处理结果 行 头 空行 体
// 1.响应行
// console.log(xhr.status)//状态码
// console.log(xhr.statusText)//状态字符串
// console.log(xhr.getAllResponseHeaders())//获取所有响应头
// console.log(xhr.response)//响应体
// 设置result的文本
result.innerHTML=xhr.response
}else{
}
}
}
})
</script>
</body>
</html>
在AJAX请求中如何设置url参数
在url地址后面添加?参数=参数值,多个参数中间用&隔开
http://127.0.0.1:8000/server?a=100&b=200&c=300
AJAX POST请求
向8000端口发送POST请求,原来的js代码只有get请求,故还要添加POST请求代码才可
app.post('/server',(request,response)=>{
// 设置响应头 设置允许跨域
response.setHeader('Access-Control-Allow-Origin','*')
// 设置响应体
response.send('HELLO AJAX POST')
})
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>AJAXPOST请求</title>
<style>
#result {
width: 200px;
height: 100px;
border: solid 1px #903;
}
</style>
</head>
<body>
<div id="result"></div>
<script>
// 获取元素对象
const result=document.getElementById('result')
result.addEventListener('mouseover',function(){
// console.log("test")
// 1.创建对象
const xhr=new XMLHttpRequest()
// 2.初始化 设置类型与URL
xhr.open('POST','http://127.0.0.1:8000/server')
// 3.发送
xhr.send()
// 4.事件绑定
xhr.onreadystatechange=function(){
if(xhr.readyState===4){
if(xhr.status>=200&&xhr.status<300){
// 处理服务器返回的结果
result.innerHTML=xhr.response
}
}
}
})
</script>
</body>
</html>
//1.引入express
const express=require('express')
// 2.创建应用对象
const app=express()
// 3.创建路由规则
// request 是对请求报文的封装
// response 是对响应报文的封装
// 当客户端浏览器向服务端发送请求时,请求行的第二段路径是/server的话,就会执行回调函数里面的代码,并且由response来作出响应
app.get('/server',(request,response)=>{
// 设置响应头 设置允许跨域
response.setHeader('Access-Control-Allow-Origin','*')
// 设置响应体
response.send('HELLO AJAX')
})
app.post('/server',(request,response)=>{
// 设置响应头 设置允许跨域
response.setHeader('Access-Control-Allow-Origin','*')
// 设置响应体
response.send('HELLO AJAX POST')
})
// 4.监听端口启动服务
app.listen(8000,()=>{
console.log("服务已经启动,8000端口监听中.....")
})
POST传递参数
在send中传入参数
xhr.send('a=100&b=200&c=300')或xhr.send('a:100&b:200&c:300')
在AJAX中设置请求头信息
在初始化open之后添加如下代码
// Content-Type设置请求体内容的类型
// application/x-www-form-urllencoded参数查询字符串类型
xhr.setRequestHeader('Content-Type','application/x-www-form-urllencoded')
设置自定义请求头
app.all('/server',(request,response)=>{
// 设置响应头 设置允许跨域
response.setHeader('Access-Control-Allow-Origin','*')
// 响应头
response.setHeader('Access-Control-Allow-Headers','*') // '*'表示所有头信息都可以接受
// 设置响应体
response.send('HELLO AJAX POST')
})
将post改为all,表示可以接收任意类型的请求(get post options等)
xhr.setRequestHeader('name','atguigu')
实现POST请求完整代码:
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>AJAXPOST请求</title>
<style>
#result {
width: 200px;
height: 100px;
border: solid 1px #903;
}
</style>
</head>
<body>
<div id="result"></div>
<script>
// 获取元素对象
const result=document.getElementById('result')
result.addEventListener('mouseover',function(){
// console.log("test")
// 1.创建对象
const xhr=new XMLHttpRequest()
// 2.初始化 设置类型与URL
xhr.open('POST','http://127.0.0.1:8000/server')
// 设置请求头
// Content-Type设置请求体内容的类型
// application/x-www-form-urllencoded参数查询字符串类型
xhr.setRequestHeader('Content-Type','application/x-www-form-urlencoded')
xhr.setRequestHeader('name','atguigu')
// 3.发送
xhr.send('a=100&b=200&c=300')
// xhr.send('a:100&b:200&c:300')
// 4.事件绑定
xhr.onreadystatechange=function(){
if(xhr.readyState===4){
if(xhr.status>=200&&xhr.status<300){
// 处理服务器返回的结果
result.innerHTML=xhr.response
}
}
}
})
</script>
</body>
</html>
//1.引入express
const express=require('express')
// 2.创建应用对象
const app=express()
// 3.创建路由规则
// request 是对请求报文的封装
// response 是对响应报文的封装
// 当客户端浏览器向服务端发送请求时,请求行的第二段路径是/server的话,就会执行回调函数里面的代码,并且由response来作出响应
app.get('/server',(request,response)=>{
// 设置响应头 设置允许跨域
response.setHeader('Access-Control-Allow-Origin','*')
// 设置响应体
response.send('HELLO AJAX')
})
// all表示可以接收任意类型的请求(get post options等)
app.all('/server',(request,response)=>{
// 设置响应头 设置允许跨域
response.setHeader('Access-Control-Allow-Origin','*')
// 响应头
response.setHeader('Access-Control-Allow-Headers','*') // '*'表示所有头信息都可以接受
// 设置响应体
response.send('HELLO AJAX POST')
})
// 4.监听端口启动服务
app.listen(8000,()=>{
console.log("服务已经启动,8000端口监听中.....")
})
JSON数据的响应以及前端代码的处理
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Document</title>
<style>
#result{
width: 200px;
height: 100px;
border: solid 1px #89b;
}
</style>
</head>
<body>
<div id="result"></div>
<script>
const result=document.getElementById('result')
// 绑定键盘按下事件
window.onkeydown=function(){
// 发送请求
const xhr=new XMLHttpRequest()
// 设置响应体数据的类型
xhr.responseType='json'
// 初始化
xhr.open('GET','http://127.0.0.1:8000/json-server')
xhr.send()
xhr.onreadystatechange=function(){
if(xhr.readyState===4){
if(xhr.status>=200&&xhr.status<300){
// console.log(xhr.response)
// result.innerHTML=xhr.response
// 1.手动对数据进行转换
// let data=JSON.parse(xhr.response)
// console.log(data)
// 2.自动转换xhr.responseType='json'
result.innerHTML=xhr.response.name
}
}
}
}
</script>
</body>
</html>
//1.引入express
const express=require('express')
// 2.创建应用对象
const app=express()
// 3.创建路由规则
// request 是对请求报文的封装
// response 是对响应报文的封装
// 当客户端浏览器向服务端发送请求时,请求行的第二段路径是/server的话,就会执行回调函数里面的代码,并且由response来作出响应
app.get('/server',(request,response)=>{
// 设置响应头 设置允许跨域
response.setHeader('Access-Control-Allow-Origin','*')
// 设置响应体
response.send('HELLO AJAX')
})
// all表示可以接收任意类型的请求(get post options等)
app.all('/json-server',(request,response)=>{
// 设置响应头 设置允许跨域
response.setHeader('Access-Control-Allow-Origin','*')
// 响应头
response.setHeader('Access-Control-Allow-Headers','*') // '*'表示所有头信息都可以接受
// 响应一个数据
const data={
name:'atguigu'
}
// 对对象进行字符串转换
let str=JSON.stringify(data)
// 设置响应体
response.send(str)//里面只能接收字符串或者buffer(保存二进制文件数据)
})
// 4.监听端口启动服务
app.listen(8000,()=>{
console.log("服务已经启动,8000端口监听中.....")
})
nodemon重新启动
在终端运行服务器代码时输入nodemon.cmd xxx.js,当服务器js代码改变时,会自动重新启动,不用手动启动
IE浏览器缓存问题
IE浏览器会对AJAX请求结果进行缓存,在下次进行请求时是从本地缓存中获取,而不是服务器返回的最新数据,对于时效性比较强的场景,AJAX缓存不能及时显示,会影响结果
解决方法:
xhr.open('GET','http://127.0.0.1:8000/ie?t='+Date.now())
请求超时与网络异常
项目向服务器请求时,不能保证服务端能够及时快速的响应,这时就会请求超时,同时,在网络异常时,也需要给用户一个提醒
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>IE缓存问题</title>
<style>
#result {
width: 200px;
height: 100px;
border: solid 1px #258;
}
</style>
</head>
<body>
<button>点击发送请求</button>
<div id="result"></div>
<script>
const btn=document.getElementsByTagName('button')[0]
const result=document.querySelector('#result')
btn.addEventListener('click',function(){
// console.log('test')
const xhr=new XMLHttpRequest()
// 超时设置2s设置
xhr.timeout=2000//2s之后若没有成功返回结果就取消
// 超时回调
xhr.ontimeout=function(){
alert("网络异常,请稍后重试!!")
}
// 网络异常回调
xhr.onerror=function(){
alert('你的网络似乎出了一些问题!')
}
xhr.open('GET','http://127.0.0.1:8000/delay')
xhr.send()
xhr.onreadystatechange=function(){
if(xhr.readyState===4){
if(xhr.status>=200&&xhr.status<300){
result.innerHTML=xhr.response
}
}
}
})
</script>
</body>
</html>
app.get('/delay',(request,response)=>{
// 设置响应头 设置允许跨域
response.setHeader('Access-Control-Allow-Origin','*')
setTimeout(()=>{
response.send('延时响应')
},3000)
// 设置响应体
})
手动取消请求
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>取消请求</title>
</head>
<body>
<button>点击发送</button>
<button>点击取消</button>
<script>
// 获取元素对象
const btns=document.querySelectorAll('button')
let x=null
btns[0].onclick=function(){
x=new XMLHttpRequest()
x.open('GET','http://127.0.0.1:8000/delay')
x.send()
}
// abort
btns[1].onclick=function(){
x.abort()
}
</script>
</body>
</html>
重复请求问题
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>重复请求</title>
</head>
<body>
<button>点击发送</button>
<button>点击取消</button>
<script>
// 获取元素对象
const btns=document.querySelectorAll('button')
let x=null
// 标识变量
let isSending=false//是否正在发送AJAX请求
btns[0].onclick=function(){
if(isSending)x.abort()//如果正在发送,则取消该请求,创建一个新的请求
x=new XMLHttpRequest()
isSending=true
x.open('GET','http://127.0.0.1:8000/delay')
x.send()
x.onreadystatechange=function(){
if(x.readyState===4){
// 修改标识变量
isSending=false
}
}
}
// abort
btns[1].onclick=function(){
x.abort()
}
</script>
</body>
</html>
jQuery中的AJAX
get请求
$.get(url,[data],[callback],[type])
url:请求的URL地址
data:请求携带的参数
callback:载入成功时回调函数
type:设置返回内容的格式,xml,html,script,json,text,_default
post请求
$.post(url,[data],[callback],[type])
url:请求的URL地址
data:请求携带的参数
callback:载入成功时回调函数
引入jquery.js文件
搜索BootCDN官网
之后搜索jquery,点击jquery后选择所需版本并复制script标签,将其放置在html里即可
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<link rel="stylesheet" href="../bootstrap-3.3.2-dist/css/bootstrap-theme.css">
<link rel="stylesheet" href="../bootstrap-3.3.2-dist/css/bootstrap.css">
<link rel="stylesheet" href="../bootstrap-3.3.2-dist/css/bootstrap.min.css">
<title>jQuery发送AJAX请求</title>
<script src="https://cdn.bootcdn.net/ajax/libs/jquery/3.5.1/jquery.js"></script>
</head>
<body>
<div class="container">
<h2 class="page-header">jQuery发送AJAX请求</h2>
<button class="btn btn-primary">GET</button>
<button class="btn btn-danger">POST</button>
<button class="btn btn-info">通用型方法ajax</button>
</div>
<script>
$('button').eq(0).click(function(){
$.get('http://127.0.0.1:8000/jquery-server',{a:100,b:200},function(data){
console.log(data)
},'json')//第一个参数给谁发,第二个为传进去的参数//第三个为回调函数,data为响应体第四个参数为响应体类型
})
$('button').eq(1).click(function(){
$.post('http://127.0.0.1:8000/jquery-server',{a:100,b:200},function(data){
console.log(data)
})//第一个参数给谁发,第二个为传进去的参数//第三个为回调函数,data为响应体
})
</script>
</body>
</html>
// jQuery服务
app.all('/jquery-server',(request,response)=>{
// 设置响应头 设置允许跨域
response.setHeader('Access-Control-Allow-Origin','*')
// response.send('Hello jQuery AJAX')
const data={name:'尚硅谷'}
response.send(JSON.stringify(data))
})
get请求第四个参数表示响应体类型,上述代码在进行get请求时,其返回结果类型为json,而post请求返回的结果只是字符串
$.ajax方法接收的参数是一个对象,包括url,参数,请求类型,成功的回调以及响应体结果的类型等等
$('button').eq(2).click(function(){
$.ajax({
//url
url:'http://127.0.0.1:8000/delay',
// 参数
data:{a:100,b:200},
// 请求类型
type:'GET',
// 响应体结果类型
dataType:'json',
// 成功的回调
success:function(data){
console.log(data)
},
// 超时时间
timeout:2000,
// 失败的回调
error:function(){
console.log('出错啦!!!')
},
// 头信息
headers:{
c:300,
d:400
}
})
})
crossorigin="anonymous"
crossorigin="anonymous"
是一个HTML属性,用于设置图像、脚本、样式表等外部资源的跨源资源共享(Cross-Origin Resource Sharing,简称 CORS)策略。当将其应用于如<img>
,<script>
,<link rel="stylesheet">
等标签时,它的作用主要是:
允许跨域加载资源:在默认情况下,浏览器实施同源策略(Same-origin policy),阻止从一个域名加载的网页脚本访问来自不同源的资源。通过设置
crossorigin="anonymous"
,你可以告诉浏览器允许请求的资源来自不同的源,并且请求应该带有一个匿名的凭据标志,表示请求不应包含 cookies 或 HTTP 认证信息。控制请求头的信息:当使用
crossorigin="anonymous"
时,浏览器会在请求头中添加一个Origin
字段,告知服务器这个请求是跨域的。服务器需要配置相应的CORS headers(如Access-Control-Allow-Origin
)来允许这种跨域请求。如果服务器允许,它将在响应头中返回Access-Control-Allow-Origin
,从而使得浏览器能够成功接收并使用这个跨域资源。无凭据请求:特别需要注意的是,
anonymous
模式下,请求不会发送cookie或其他认证信息,这与使用withCredentials
属性发送凭据的请求(通常用于Ajax请求)不同。这对于那些不需要用户特定信息的公共资源加载非常有用,提高了安全性,因为减少了信息泄露的风险。总之,
crossorigin="anonymous"
主要用于实现安全地跨域加载脚本、样式和图像等静态资源,而不涉及用户的认证信息,是一种提高Web应用安全性和灵活性的机制。
Axios发送AJAX请求
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<script crossorigin="anonymous" src="https://cdn.bootcdn.net/ajax/libs/axios/1.6.8/axios.js"></script>
<title>axios 发送AJAX请求</title>
</head>
<body>
<button>GET</button>
<button>POST</button>
<button>AJAX</button>
<script>
const btns=document.querySelectorAll('button')
// 配置baseURL
axios.defaults.baseURL='http://127.0.0.1:8000'
btns[0].onclick=function(){
// GET请求
axios.get('/axios-server',{
// url参数
params:{
id:100,
vip:7
},
// 请求头信息
Headers:{
name:'atguigu',
age:20
}
}).then(value=>{//对返回结果进行处理,与jquery的回调函数的作用相似
console.log(value)
})
}
btns[1].onclick=function(){
axios.post('/axios-server',
// 请求体
{
username:'admin',
password:'admin'
},{
// url
params:{
id:200,
vip:9
},
// 请求头参数
Headers:{
weight:180,
height:180
}
})
}
</script>
</body>
</html>
服务端代码:
// axios服务
app.all('/axios-server',(request,response)=>{
// 设置响应头 设置允许跨域
response.setHeader('Access-Control-Allow-Headers','*') // '*'表示所有头信息都可以接受
response.setHeader('Access-Control-Allow-Origin','*')
// response.send('Hello jQuery AJAX')
const data={name:'尚硅谷'}
response.send(JSON.stringify(data))
})
axios通用方法发送请求
btns[2].onclick=function(){
axios({
// 请求方法
method:'POST',
// url
url:'/axios-server',
// url参数
params:{
vip:10,
level:30
},
// 头信息
headers:{
a:100,
b:200
},
// 请求体参数
data:{
username:'admin',
password:'admin'
}
}).then(response=>{
console.log(response)
// 响应状态码
console.log(response.status)
// 响应状态字符串
console.log(response.statusText)
// 响应头信息
console.log(response.headers)
// 响应体
console.log(response.data)
})
}
fetch发送AJAX请求
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>fetch 发送 AJAX请求</title>
</head>
<body>
<button>AJAX请求</button>
<script>
const btn=document.querySelector('button')
btn.onclick=function(){
//接收两个参数,第一个为url,第二个为可选配置对象
fetch('http://127.0.0.1:8000/fetch-server?vip=10',{
// 请求方法
method:'POST',
// 请求头
headers:{
name:'atguigu'
},
body:'username=admin&password=admin'
}).then(response=>{
// console.log(response)
// return response.text()
return response.json()//把数据转化成json对象
}).then(response=>{
console.log(response)
})
}
</script>
</body>
</html>
服务端代码:
// fetch服务
app.all('/fetch-server',(request,response)=>{
// 设置响应头 设置允许跨域
response.setHeader('Access-Control-Allow-Headers','*') // '*'表示所有头信息都可以接受
response.setHeader('Access-Control-Allow-Origin','*')
// response.send('Hello jQuery AJAX')
const data={name:'尚硅谷'}
response.send(JSON.stringify(data))
})
同源策略
同源策略是浏览器的一种安全策略
同源:当前网页的url和AJAX请求的目标资源的url的协议,域名,端口号必须完全相同
AJAX默认遵从同源策略,违背同源策略就是跨域
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>首页</title>
</head>
<body>
<h1>尚硅谷</h1>
<button>点击获取用户数据</button>
<script>
const btn=document.querySelector('button')
btn.onclick=function(){
const x=new XMLHttpRequest()
// 因为满足同源策略 所以url可以简写
x.open("GET",'/data')
// 发送
x.send()
x.onreadystatechange=function(){
if(x.readyState===4){
if(x.status>=200&&x.status<300){
console.log(x.response)
}
}
}
}
</script>
</body>
</html>
const express=require('express')
const app=express()
app.get('/home',(request,response)=>{
// 响应一个页面
response.sendFile(__dirname+'/index.html')//__dirname在node中是当前文件的绝对路径
})
app.get('/data',(request,response)=>{
response.send('用户数据')
})
app.listen(9000,()=>{
console.log("服务已经启动...")
})
在网址中输入127.0.0.1:9000/home即可进入首页页面
如何解决跨域
JSONP
JSONP是什么?
JSONP,是一个官方的跨域解决方案,纯粹凭借程序员的聪明才智开发出来,只支持get请求
JSONP是怎么工作的?
在网页有一些标签具有跨域能力,比如:img,link,iframe,script
JSONP就是利用script标签的跨域能力来发送请求的
JSONP的使用
1.动态的创建一个script标签
var script=document.creatElement("script")
2.设置script的src,设置回调函数
script.src="http://localhost:3000/testAJAX?callback=abc"
原理实现
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>原理演示</title>
<style>
#result{
width: 300px;
height: 100px;
border: solid 1px #78a;
}
</style>
</head>
<body>
<div id="result"></div>
<script>
function handle(data){
// 获取result元素
const result=document.getElementById('result')
result.innerHTML=data.name
}
</script>
<!-- <script src="./js/app.js"></script> -->
<!-- <script src="http://127.0.0.1:5500/%E8%B7%A8%E5%9F%9F/JSONP/%E5%8E%9F%E7%90%86.html"></script> -->
<script src="http://127.0.0.1:8000/jsonp-server"></script>
<!-- script发送请求应该返回标准的js代码 -->
</body>
</html>
app.js
const data={
name:'尚硅谷'
}
console.log(data)
// function handle(data){
// // 获取result元素
// const result=document.getElementById('result')
// result.innerHTML=data.name
// }
handle(data)
8000端口:
// jsonp服务
app.all('/jsonp-server',(request,response)=>{
// response.send('console.log("hello jsonp-server")')
const data={
name:'尚硅谷'
}
// 将数据转化为字符串
let str=JSON.stringify(data)
// 返回结果
response.end(`handle(${str})`)//end不会加特殊响应头
})
原生JSONP的实现
案例:用户注册输入用户名后,当光标失去焦点时,向服务端发送请求,服务端返回是否存在该用户名,存在就将input框变成红色
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>案例</title>
</head>
<body>
用户名: <input type="text" id="username">
<p></p>
<script>
// 获取input元素
const input=document.querySelector('input')
const p=document.querySelector('p')
// 声明handle函数
function handle(data){
input.style.border="solid 1px #f00"
p.innerHTML=data.msg
}
// 绑定事件
input.onblur=function(){
// 获取用户的输入值
let username=this.value
// 向服务器端发送请求 检测用户名是否存在
// 1.创建标签的src属性
const script=document.createElement('script')
// 2.设置标签的src属性
script.src='http://127.0.0.1:8000/check-username'
document.body.appendChild(script)
}
</script>
</body>
</html>
// 检测用户名是否存在
app.all('/check-username',(request,response)=>{
// response.send('console.log("hello jsonp-server")')
const data={
exist:1,
msg:'用户名已经存在'
}
// 将数据转化为字符串
let str=JSON.stringify(data)
// 返回结果
response.end(`handle(${str})`)//end不会加特殊响应头
})
用jQuery发送jsonp请求
点击按钮,向8000端口发送jsonp请求,将返回结果放置在result中
$.getJSON()
是 jQuery 库中的一个函数,用于简化 AJAX 请求,专门用于获取 JSON 格式的数据。这个函数是$.ajax()
的一个便捷封装,用于执行 HTTP GET 请求,并自动将返回的数据解析为 JavaScript 对象(即 JSON 格式数据)。使用
$.getJSON()
的基本语法如下:$.getJSON(url, data, function(data) { // 这个函数会在请求成功并且数据被解析后执行 // "data" 参数包含了从服务器返回的 JSON 数据,已经被转换成 JavaScript 对象 console.log(data); }).done(function() { // 可选的,当请求成功完成时执行 }).fail(function(jqXHR, textStatus, errorThrown) { // 可选的,当请求失败时执行 }).always(function() { // 可选的,无论请求成功或失败都会执行 });
- url: (字符串)必需,规定要发送请求的 URL。
- data: (可选,对象或字符串)发送到服务器的数据,可以是对象的键值对或查询字符串形式。
- callback: (函数)当请求成功,并且服务器返回了数据时,会执行这个函数。返回的数据会作为参数传递给这个函数。
.done()
,.fail()
,.always()
方法提供了对成功、失败和完成状态的回调处理,增加了请求的灵活性。请注意,由于 jQuery 是一个库,所以在使用
$.getJSON()
之前,确保已经在项目中引入了 jQuery。
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>jQuery-jsonp</title>
<style>
#result {
width: 300px;
height: 100px;
border: solid 1px #089;
}
</style>
<script crossorigin="anonymous" src="https://cdn.bootcdn.net/ajax/libs/jquery/3.5.1/jquery.min.js"></script>
</head>
<body>
<button>点击发送jsonp请求</button>
<div id="result">
</div>
<script>
$('button').eq(0).click(function(){
$.getJSON('http://127.0.0.1:8000/jquery-jsonp-server?callback=?',function(data){
// console.log(data)
$('#result').html(
`
名称:${data.name}<br>
校区:${data.city}
`
)
})
})
</script>
</body>
</html>
// 用jQuery发送jsonp请求
app.all('/jquery-jsonp-server',(request,response)=>{
// response.send('console.log("hello jsonp-server")')
const data={
name:'尚硅谷',
city:['北京','上海','深圳']
}
// 将数据转化为字符串
let str=JSON.stringify(data)
// 接收callback参数
let cb=request.query.callback//接收callback返回的参数
// 返回结果
response.end(`${cb}(${str})`)//end不会加特殊响应头
})
CORS
CORS是什么?
CORS,跨域资源共享,CORS是官方的跨域解决方案,它的特点是不需要在客户端做任何特殊的操作,完全在服务器中进行处理,支持get和post请求,跨域资源共享标准新增了一组HTTP首部字段,允许服务器声明哪些源站通过浏览器有权访问哪些资源
CORS是怎么工作的
CORS是通过设置一个响应头来告诉浏览器,该请求允许跨域,浏览器收到该响应以后就会对响应放行
CORS的使用
主要是服务器端的设置:
router.get("/testAJAX",function(req,res){})
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Document</title>
<style>
#result {
width: 200px;
height: 100px;
border: solid 1px #90b;
}
</style>
</head>
<body>
<button>发送请求</button>
<div id="result"></div>
<script>
const btn=document.querySelector('button')
btn.onclick=function(){
// 1.创建对象
const x=new XMLHttpRequest()
// 2.初始化位置
x.open("GET",'http://127.0.0.1:8000/cors-server')
// 3.发送
x.send()
// 4.绑定事件
x.onreadystatechange=function(){
if(x.readyState===4){
if(x.status>=200&&x.status<300){
// 输出响应体
console.log(x,response)
}
}
}
}
</script>
</body>
</html>
// cros
app.all('/cors-server',(request,response)=>{
//设置响应头
response.setHeader('Access-Control-Allow-Origin','*')
response.setHeader('Access-Control-Allow-Headers','*')
response.setHeader('Access-Control-Allow-Method','*')//客户端在发送请求时,跨域页面哪个都行,头信息和请求方法都随意
// response.setHeader('Access-Control-Allow-Orign','http://127.0.0.1:5500')//只有http://只有127.0.0.1:5500端口的网页才可向该服务器发生请求
response.send('hello CORS')
})