AJAX知识点(前后端交互技术)

原生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的优点

  1. 可以无需刷新页面而与服务器端进行通信
  2. 允许你根据用户事件来更新部分页面内容

AJAX的缺点

  1. 没有浏览历史,不能回退
  2. 存在跨域问题(同源)
  3. 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"> 等标签时,它的作用主要是:

  1. 允许跨域加载资源:在默认情况下,浏览器实施同源策略(Same-origin policy),阻止从一个域名加载的网页脚本访问来自不同源的资源。通过设置 crossorigin="anonymous",你可以告诉浏览器允许请求的资源来自不同的源,并且请求应该带有一个匿名的凭据标志,表示请求不应包含 cookies 或 HTTP 认证信息。

  2. 控制请求头的信息:当使用 crossorigin="anonymous" 时,浏览器会在请求头中添加一个 Origin 字段,告知服务器这个请求是跨域的。服务器需要配置相应的CORS headers(如 Access-Control-Allow-Origin)来允许这种跨域请求。如果服务器允许,它将在响应头中返回 Access-Control-Allow-Origin,从而使得浏览器能够成功接收并使用这个跨域资源。

  3. 无凭据请求:特别需要注意的是,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')
})

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/608439.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

【进程等待】阻塞等待 | options非阻塞等待

目录 waitpid 阻塞等待 options&非阻塞等待 pid_t返回值 阻塞等待VS非阻塞等待 waitpid 回顾上篇&#xff1a; pid_ t waitpid(pid_t pid, int *status, int options); 返回值&#xff1a; 当正常返回的时候waitpid返回收集到的子进程的进程ID&#xff1b;如果设置了…

C++容器之vector类

目录 1.vector的介绍及使用1.1vector的介绍1.2vector的使用1.2.1 vector的定义1.2.2 vector iterator 的使用1.2.3 vector 空间增长问题1.2.4 vector 增删查改1.2.5vector 迭代器失效问题1.2.6 vector 在OJ中的使用。 2.vector深度剖析及模拟实现2.1 std::vector的核心框架接口…

金三银四面试题(二十五):策略模式知多少?

什么是策略模式 策略模式&#xff08;Strategy Pattern&#xff09;是一种行为型设计模式&#xff0c;旨在定义一系列算法&#xff0c;将每个算法封装到一个独立的类中&#xff0c;使它们可以互换。策略模式让算法的变化独立于使用它们的客户端&#xff0c;使得客户端可以根据…

车载测试系列:入行车载测试分享

车载测试前景如何&#xff1f; 软件定义汽车时代的发展趋势&#xff0c;随着控制器自主开发力度的加强&#xff0c;作为V流程中必备环节&#xff0c;车载测试工程师岗位需求会越来越多&#xff1b;控制器集成化&#xff0c;功能集成程度越来越高&#xff0c;对于测试工程师的知…

3. 初探MPI——(非阻塞)点对点通信

系列文章目录 初探MPI——MPI简介初探MPI——&#xff08;阻塞&#xff09;点对点通信初探MPI——&#xff08;非阻塞&#xff09;点对点通信初探MPI——集体通信 文章目录 系列文章目录前言一、Non-blocking communications1.1 Block version1.2 Non-blocking version 二、准…

思维导图软件哪个好?盘点这5款好用的工具!

思维导图作为一种有效的思维工具&#xff0c;在日常生活和工作中扮演着越来越重要的角色。无论是学习、工作规划&#xff0c;还是项目管理&#xff0c;思维导图都能帮助我们更好地组织思路&#xff0c;提升工作效率。然而&#xff0c;市面上众多的思维导图软件让人眼花缭乱&…

软件系统工程建设全套资料(交付清单)

软件全套精华资料包清单部分文件列表&#xff1a; 工作安排任务书&#xff0c;可行性分析报告&#xff0c;立项申请审批表&#xff0c;产品需求规格说明书&#xff0c;需求调研计划&#xff0c;用户需求调查单&#xff0c;用户需求说明书&#xff0c;概要设计说明书&#xff0c…

C++类和对象(4)

目录 1.初始化列表 2.单参数里面的隐式类型转换 3.多参数的隐式类型转换 4.匿名对象 1.初始化列表 &#xff08;1&#xff09;首先看一下初始化列表具体是什么&#xff1f; 这个就是初始化列表的具体形式&#xff0c;对&#xff0c;你没有看错&#xff0c;这个初始化列表里…

python:画折线图

import pandas as pd import matplotlib.pyplot as plt from matplotlib.font_manager import FontProperties# 设置新宋体字体的路径 font_path D:/reportlab/simsun/simsun.ttf# 加载新宋体字体 prop FontProperties(fnamefont_path)""" # 读取 xlsx 文件 d…

【投资必看】充电桩加盟合作哪家好,充电桩厂家合作模式一般有哪些?

随着新能源汽车行业的蓬勃发展&#xff0c;充电桩作为关键的基础设施&#xff0c;其市场需求日益增长。对于有意进入这一行业的投资者来说&#xff0c;了解和选择适合的合作模式至关重要。充电桩厂家的合作模式一般有哪些&#xff0c;本文将从设备销售和投资运营两个维度进行讨…

容灾演练双月报|郑大一附院数据级容灾演练切换

了解更多灾备行业动态 守护数字化时代业务连续 目录 CONTENTS 01 灾备法规政策 02 热点安全事件 03 容灾演练典型案例 01 灾备法规政策 3月19日&#xff0c;工信部发布《工业和信息化部办公厅关于做好2024年信息通信业安全生产和网络运行安全工作的通知》。明确提出“…

官宣:vAsterNOS正式发布!开放网络操作系统免费试用!

近期&#xff0c;vAsterNOS&#xff08;设备模拟器&#xff09;正式发布&#xff0c;可以满足用户快速了解 AsterNOS、体验实际操作、搭建模拟网络的需求&#xff0c;可运行在GNS3、EVE-NG等网络虚拟软件中。 AsterNOS 网络操作系统是星融元为人工智能、机器学习、高性能计算、…

java培训班还值得去培训吗?

请大家关注我的公众号&#xff1a;老胡聊Java 1 应届生或者在校生&#xff0c;如果感觉有必要&#xff0c;可以去提升下技术&#xff0c;因为应届生或在校生找工作时&#xff0c;未必要提升真实项目经验&#xff0c;所以用应届生身份学到的spring boot等java技术背面试题&#…

《二十三》Qt 简单小项目---视频播放器

QT 使用QMediaPlayer实现的简易视频播放器 效果如下&#xff1a; 功能点 播放指定视频点击屏幕暂停/播放开始/暂停/重置视频拖拽到指定位置播放 类介绍 需要在配置文件中加入Multimedia, MultimediaWidgets这俩个库。 Multimedia&#xff1a;提供了一套用于处理音频、视频…

如何开启深色模式【攻略】

如何开启深色模式【攻略】 前言版权推荐如何开启深色模式介绍手机系统手机微信手机QQ手机快手手机抖音 电脑系统电脑微信电脑QQ电脑WPS电脑浏览器 最后 前言 2024-5-9 20:48:21 深色模式给人以一种高级感。 本文介绍一些常用软件深色模式的开启 以下内容源自《【攻略】》 仅…

基于Spring Boot的酒店管理系统设计与实现

基于Spring Boot的酒店管理系统设计与实现 开发语言&#xff1a;Java 框架&#xff1a;springboot JDK版本&#xff1a;JDK1.8 数据库工具&#xff1a;Navicat11 开发软件&#xff1a;eclipse/myeclipse/idea 系统部分展示 系统首页界面图&#xff0c;在系统首页可以查看首页…

【数据结构-二叉搜索树的增删查改】

&#x1f308;个人主页&#xff1a;努力学编程’ ⛅个人推荐&#xff1a;基于java提供的ArrayList实现的扑克牌游戏 |C贪吃蛇详解 ⚡学好数据结构&#xff0c;刷题刻不容缓&#xff1a;点击一起刷题 &#x1f319;心灵鸡汤&#xff1a;总有人要赢&#xff0c;为什么不能是我呢 …

python-类和对象

1、设计一个 Circle类来表示圆,这个类包含圆的半径以及求面积和周长的函数。再使用这个类创建半径为1~10的圆,并计算出相应的面积和周长。 &#xff08;1&#xff09;源代码&#xff1a; import math class Circle: def __init__(self, r): self.r r #面积 def area(self): r…

最佳实践 | 八爪鱼采集器如何用PartnerShare做全民分销?

在数字化时代&#xff0c;数据采集和分析已经成为企业运营和决策的重要一环。八爪鱼采集器作为一款领先的SaaS产品&#xff0c;凭借其强大的数据采集和处理能力&#xff0c;成为了众多企业和个人用户的心头好。为了进一步拓展市场份额&#xff0c;提升品牌影响力&#xff0c;八…

TCP通信并发:

上次的程序只能保持&#xff0c;单线程或者进程 多进程并发服务器 进程的特点&#xff08;有血缘关系&#xff09; 创建子进程&#xff1a;fork&#xff08;&#xff09;&#xff1b; 虚拟地址空间被复制 &#xff0c;从一份变成两份&#xff08;用户区和内核区&#xff09…