【Node.js】 Express编写接口和跨域解决的方法!

news2024/9/25 9:38:45

Node.js

        • Express编写接口
        • 接口跨域问题
          • 使用cors中间件解决跨域问题
          • CORS响应头分类:
          • CORS请求分类:
        • 编写JSONP接口

Express编写接口

  • 注意想获取URl-encoded格式的请求体数据,必须配置中间件app.use(express.urlencoded({extended:false}))
  • 创建api路由模块,编写接口
// 导入express
const express = require('express')
// 创建路由对象
const router = express.Router()
// 编写接口
router.get('/get', (req, res) => {
    // 1.获取客户端通过查询字符串,发送到服务器的数据
    console.log(req.query)
    const query = req.query
    // 2.调用res.send()方法,把数据返回给客户端
    res.send({
        code:1,
        msg:"请求成功",
        date:query
    })
})

router.post('/post', (req, res) => {
    // 1.获取客户端通过查询字符串,发送到服务器的数据
    console.log(req.body)
    const body = req.body
    // 2.调用res.send()方法,把数据返回给客户端
    res.send({
        code:1,
        msg:"POST请求成功",
        date:body
    })
})
//向外共享路由对象
module.exports = router
  • 导入路由模块,运行服务器
// 导入express
const express = require('express')
// 创建web服务器
const app = express()
// 启动文本服务器
app.listen(80, () => {
    console.log("80服务器启动http://127.0.0.1/")
})
//配置解析表单数据的中间件,否则post接收的URl-encoded格式的请求体数据为空
app.use(express.urlencoded({extended:false}))
// 导入路由模块
const router = require('./router')
// 全局注册,加前缀
app.use('/api', router)

  • 调用图片,注意地址
    在这里插入图片描述
    在这里插入图片描述

接口跨域问题

  • 上面的GET和POST接口。存在不支持跨域请求的问题
<!DOCTYPE html>
<html lang="en">

<head>
    <meta charset="UTF-8">
    <meta http-equiv="X-UA-Compatible" content="IE=edge">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>Document</title>
</head>
<script src="https://code.jquery.com/jquery-3.4.1.min.js"></script>

<body>
    <button id="btnGET">跨域请求GET</button>
    <button id="btnPOSt">跨域请求POSt</button>
    <script>
        // 1,测试GET接口
        $('#btnGET').on('click', () => {
            $.ajax({
                type: 'Get',
                url: 'http://127.0.0.1/api/get',
                data: {
                    "name": "乞力马扎罗GET",
                    "age": "24"
                }
            })
        })
         // 2,测试POST接口
         $('#btnPOSt').on('click', () => {
            $.ajax({
                type: 'POST',
                url: 'http://127.0.0.1/api/post',
                data: {
                    "name": "乞力马扎罗POST",
                    "age": "24"
                }
            })
        })
    </script>
</body>

</html>
  • 报错图片

在这里插入图片描述

  • 解决方法
  • CORS(主流的解决方案,推荐使用)
  • JSONP(有缺陷的解决方案,只支持GET请求)
使用cors中间件解决跨域问题
  • cors是Express的一个第三方中间件,通过安装和配置中间件,可以很方便的解决跨域问题
  • 运行npm install cors 安装中间件
  • 使用const cors =require(“cors”)导入中间件
  • 在路由之前调用app.use(cors())配置中间件
  • 一定要在路由之前配置这个中间件
//导入cors跨域中间件
const cors =require("cors")
// 全局注册,加前缀
app.use(cors())

什么是CORS

  • CORS,跨域资源共享,由一系列HTTP响应头组成,这些HTTP响应头决定游览器是否阻止前端js代码跨域获取资源
  • 游览器的同源安全策略默认会阻止网页的“跨域”获取资源,但如果接口服务器配置了CORS相关的响应头,就可以解除游览器端的跨域访问限制
  • 注意点。cors主要在服务器端进行配置,客户端无需进行额外的配置,即可请求开启cors的接口
  • CORS在游览器中由兼容性,只有支持XMLHTTP Request Level2的游览器,才能正常访问开启CORS的服务端接口(例如,IE10+,Chrome4+ ,fireFox3.5+)
CORS响应头分类:

Access-Control-Allow-Origin

  • 响应头部中可以携带一个Access-Control-Allow-Origin字段,其语法如下
 Access-Control-Allow-Origin:<origin>|*
  • origin参数的值指定了允许访问资源的外域URL
  • 例如:下面的字段值允许访问来自http://itcast.cn的请求
 res.setHeader('Access-Control-Allow-Origin',"http://itcast.cn")
 //*号代表通配符,表示允许来自任何域的请求
  res.setHeader('Access-Control-Allow-Origin',"*")

Access-Control-Allow-Headers

  • 默认情况下,cors仅支持客户端向服务器发送如下的9个请求头
  • Accept
  • Accept-Language
  • Content-Language
  • DPR
  • DownLink
  • Save-Date
  • Viewport-Width
  • Width
  • Content-Type(值仅限text/plain,multipart/form-data,application/x-www-form-urlencoded三者之一)
  • 如果客户端向服务器发送了额外的请求头信息,则需要在服务器端,通过Access-Control-Allow-Headers对额外的请求头进行声明,否则请求会失败
res.setHeader('Access-Control-Allow-Headers',"Content-Type,X-Custom-Header")

Access-Control-Allow-Methods

  • 默认情况下,cors仅支持客户端发起GET,POST,HEAD请求
  • 如果客户端希望通过PUT,DELETXE等方式请求服务器的资源,则需要在服务器端,通过Access-Control-Allow-Methods来指明实际请求头所允许使用的请求头
res.setHeader('Access-Control-Allow-Methods',"POST,GET,DELETE,HEAD")
//允许所有的HTTP请求方式
res.setHeader('Access-Control-Allow-Methods',"*")
CORS请求分类:
  • 客户端在请求CORS接口时候,根据请求方式和请求头的不同,可以将CORS请求分为两大类

简单请求

  • 当请求方式为GET,POST,HEAD三者之一
  • HTTP请求头部信息不超过那上面的九种,且无自定义头部字段
  • 客户端和服务器之间只会发生一次请求

预检请求

  • 当请求方式为GET,POST,HEAD三者之外的请求、
  • HTTP请求头部信息包含自定义头部字段
  • 向服务器发送了application/json格式的数据
  • 在游览器和服务器正式通信之前,游览器会先发送OPTION请求进行预检,以获知服务器是否允许该实际请求,所以这一次的OPTION请求称为”预检请求“
  • 服务器成功请求后,才会发送真正的请求,并且携带真实数据
  • 客户端和服务器之间会发生两次请求,预检请求成功之后,才会发起真正的请求
<script src="https://code.jquery.com/jquery-3.4.1.min.js"></script>
<body>
 		<button id="btnDEL">跨域请求POSt</button>
 		  // 2,测试预检接口
         $('#btnDEL').on('click', () => {
            $.ajax({
                type: 'DELETE',
                url: 'http://127.0.0.1/api/delete',
                data: {
                    "name": "乞力马扎罗POST",
                    "age": "24"
                },
                success:function(res){
                    console.log(res)
                }
            })
        })
 </body>
router.delete('/delete', (req, res) => {
    // 1.获取客户端通过查询字符串,发送到服务器的数据
    console.log(req.body)
    const body = req.body
    // 2.调用res.send()方法,把数据返回给客户端
    res.send({
        code:1,
        msg:"delete请求成功",
        date:body
    })
})
  • 会执行两次请求,第一次预检请求,总共两次请求
    在这里插入图片描述

编写JSONP接口

  • 游览器端通过<script标签的src属性,请求服务器上的数据,同时,服务器返回一个函数的调用

特点:

  • JSONP不属于真正的Ajax请求,因为它没有使用XMLHttoRrquest这个对象
  • JSONP仅支持GET请求,不支持其他请求

注意点:

  • 当项目中已经配置了CORS跨域资源共享,为了防止冲突,必须在配置CORS中间件之前声明JSONP的接口,否则会被处理成开启CORS的接口

编写jsonp

  • jsonp接口配置,一定在cors跨域中间件之前配置
  • dataType:“jsonp”,,这个在请求的时候要必须指定,否则报错
  • 就是在客户端那里建一个函数,然后在服务端将数据以通过函数调用的方式返回给客户端
  • 步骤:
  • 获取客户端发送过来的回调函数的名字
  • 得到要通过JSOP形式发送给客户的数据
  • 根据前两步得到的数据,拼接出一个函数调用的字符串
  • 把上一步得到的字符串,响应给客户端的<jsonp标签进行解析执行
   <button id="btnjsonp">jsonp请求</button>
<script>
 // 4,实现jsonp接口的具体代码
          $('#btnjsonp').on('click', () => {
            $.ajax({ 
                type: 'get',
                url: 'http://127.0.0.1/api/jsonp',
                data: {
                    "name": "乞力马扎罗POST",
                    "age": "24"
                },
                dataType:"jsonp",
                success:function(res){
                    console.log(res)
                }
            })
        })
</script>
// 导入express
const express = require('express')
// 创建web服务器
const app = express()
// 启动文本服务器
app.listen(80, () => {
    console.log("80服务器启动http://127.0.0.1/")
})
//配置解析表单数据的中间件
app.use(express.urlencoded({
    extended: false
}))
//jsonp接口配置,一定在cors跨域中间件之前配置
app.get('/api/jsonp', (req, res) => {
    // 定义JSONP的接口的实现过程
    // 1.获取客户端发送过来的回调函数的名字
    const backname = req.query.callback
    // 2.得到要通过jsonp形式发送的数据
    const data = {
        name: "乞力马扎罗",
        age: "22"
    }
    // 3.拼接出一个函数调用的字符串
    const scrStr=`${backname}(${JSON.stringify(data)})`
    // 4.得到的字符串,响应给客户端
    res.send(scrStr)
})
//导入cors跨域中间件
const cors = require("cors")
// 全局注册,加前缀
app.use(cors())
// 导入路由模块
const router = require('./router')
// 全局注册,加前缀
app.use('/api', router)
  • jsonp结果
    在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

动态规划:leetcode 1049.最后一块石头的重量II、494.目标和、474.一和零

leetcode 1049.最后一块石头的重量IIleetcode 494.目标和leetcode 474.一和零leetcode 1049.最后一块石头的重量II有一堆石头&#xff0c;每块石头的重量都是正整数。每一回合&#xff0c;从中选出任意两块石头&#xff0c;然后将它们一起粉碎。假设石头的重量分别为 x 和 y&am…

【算法经典题集】递归(持续更新~~~)

&#x1f63d;PREFACE&#x1f381;欢迎各位→点赞&#x1f44d; 收藏⭐ 评论&#x1f4dd;&#x1f4e2;系列专栏&#xff1a;算法经典题集&#x1f50a;本专栏涉及到的知识点或者题目是算法专栏的补充与应用&#x1f4aa;种一棵树最好是十年前其次是现在1.递归1.1 递归实现…

【JAVA】常见的七大排序

前言&#xff1a;本篇主要介绍常见的七大排序&#xff0c;实现语言为Java&#xff0c;其主要分为&#xff1a;直接插入排序&#xff0c;希尔排序&#xff0c;直接选择排序&#xff0c;堆排序&#xff0c;冒泡排序&#xff0c;快速排序&#xff0c;归并排序。 在介绍七大排序之前…

【多尺度密集递归融合网络:超分】

A novel image super-resolution algorithm based on multi-scale dense recursive fusion network &#xff08;基于多尺度密集递归融合网络的图像超分辨率算法&#xff09; 随着卷积神经网络(CNN)技术的成熟度,超限分辨的图像重建(SR)方法基于CNN正在蓬勃发展,取得了许多显…

RIP路由协议的更新(电子科技大学TCP/IP第二次实验)

一&#xff0e;实验目的 1、掌握 RIP 协议在路由更新时的发送信息和发送方式 2、掌握 RIP 协议的路由更新算法 二&#xff0e;预备知识 1、静态路由选择和动态路由选择 2、内部网关协议和外部网关协议 3、距离向量路由选择 三&#xff0e;实验原理 RIP 协议&#xff08…

使用Python进行数据分析——线性回归分析

大家好&#xff0c;线性回归是确定两种或两种以上变量之间互相依赖的定量关系的一种统计分析方法。根据自变量的个数&#xff0c;可以将线性回归分为一元线性回归和多元线性回归分析。一元线性回归&#xff1a;就是只包含一个自变量&#xff0c;且该自变量与因变量之间的关系是…

AMBA-AXI(一)burst 传输-INCR/WRAP/Fixed

&#x1f4a1;Note&#xff1a;本文是根据AXI协议IHI0022F_b_amba_axi_protocol_spec.pdf&#xff08;issue F&#xff09;整理的。主要是分享AXI3.0和4.0部分。如果内容有问题请大家在评论区中指出&#xff0c;有补充或者疑问也可以发在评论区&#xff0c;互相学习&#x1f64…

JUC 体系的基石——AQS

—— AQS&#xff08;AbstractQueuedSynchronizer&#xff09; 概念 抽象队列同步器&#xff1b;volatile cas 机制实现的锁模板&#xff0c;保证了代码的同步性和可见性&#xff0c;而 AQS 封装了线程阻塞等待挂起&#xff0c;解锁唤醒其他线程的逻辑。AQS 子类只需要根据状…

182、【动态规划/数组】leetcode ——647. 回文子串:动态规划+双指针(C++版本)

题目描述 原题链接&#xff1a;647. 回文子串 解题思路 &#xff08;1&#xff09;动态规划 动态规划的思路是每次判定子串两端对称位置是否相等&#xff0c;然后再基于已有的内侧对称情况判定是否为回文串。 动态规划五步曲&#xff1a; &#xff08;1&#xff09;dp[i][…

语音识别与Python编程实践

博主简介 博主是一名大二学生&#xff0c;主攻人工智能研究。感谢让我们在CSDN相遇&#xff0c;博主致力于在这里分享关于人工智能&#xff0c;c&#xff0c;Python&#xff0c;爬虫等方面知识的分享。 如果有需要的小伙伴可以关注博主&#xff0c;博主会继续更新的&#xff0c…

uni-app入门并使用学习

笔记课程 工具准备 下载HBuilderX 点击下载HBuilderX 下载微信开发者工具 点击下载微信开发者工具 使用参考uni-app官网 官网 新建项目运行 文件---新建----项目 运行到谷歌浏览器H5 运行------谷歌浏览器打开---打开成功&#xff08;第一次可能需要安装插件&#xff0…

React(三) ——新、旧生命周期

&#x1f9c1;个人主页&#xff1a;个人主页 ✌支持我 &#xff1a;点赞&#x1f44d;收藏&#x1f33c;关注&#x1f9e1; 文章目录⛳React生命周期&#x1f30b;初始化阶段&#x1f463;运行中阶段&#x1f3d3;销毁阶段&#x1f3eb;新生命周期的替代&#x1f69a;react中性…

MS9123是一款单芯片USB投屏器,内部集成了USB2 0控制器和数据收发模块、视频DAC和音视频处理模块,MS9123可以通过USB接口显示或者扩展PC、

MS9123是一款单芯片USB投屏器&#xff0c;内部集成了USB2.0控制器和数据收发模块、视频DAC和音视频处理模块&#xff0c;MS9123可以通过USB接口显示或者扩展PC、智能手机、平板电脑的显示信息到更大尺寸的显示设备上&#xff0c;支持CVBS、S-Video视频接口。 主要功能特征 C…

基本中型网络的仿真(RYU+Mininet的SDN架构)-以校园为例

目录 ​​​​​​​具体问题可以私聊博主 一、设计目标 1.1应用场景介绍 1.2应用场景设计要求 网络配置方式 网络技术要求 网络拓扑要求 互联互通 二、课程设计内容与原理 &#xff08;1&#xff09;预期网络拓扑结构和功能 &#xff08;1&#xff09;网络设备信息 …

aws ecr 使用golang实现的简单镜像转换工具

https://pkg.go.dev/github.com/docker/docker/client#section-readme 通过golang实现一个简单的镜像下载工具 总体步骤 启动一台海外区域的ec2实例安装docker和awscli配置凭证访问国内ecr仓库编写web服务实现镜像转换和自动推送 安装docker和awscli sudo yum remove awsc…

超市怎么做微信小程序_线上超市小程序开发可以实现什么功能呢

1。开发超市小程序有什么价值&#xff1f; 1、对于消费者来说&#xff1a;通过超市小程序能够更加直接的购买到想要的产品&#xff0c;消费者无需再到门店寻找商品可以直接通过超市小程序进行在线浏览&#xff1b;通过在线搜索的方式能够更加便捷的搜索到相应的商品&#xff0…

第一篇自我介绍(单片机)

小白的单片机之旅 &#x1f914;自我介绍&#x1f914; &#x1f60a;学习目标&#x1f60a; &#x1f61c;关于单片机&#x1f61c; &#x1f31d;小结&#x1f31d; &#x1f389;博客主页&#xff1a;小智_x0___0x_ &#x1f389;欢迎关注&#xff1a;&#x1f44d;点赞&…

JavaSE学习笔记day14

二、Set Set集合是Collection集合的子接口,该集合中不能有重复元素!! Set集合提供的方法签名,与父接口Collection的方法完全一致!! 即没有关于下标操作的方法 Set接口,它有两个常用的子实现类HashSet,TreeSet 三、HashSet HashSet实现了Set接口,底层是hash表(实际上底层是HashM…

QML 中的 5 大布局

作者: 一去、二三里 个人微信号: iwaleon 微信公众号: 高效程序员 在 QML 中,可以通过多种方式对元素进行布局 - 手动定位、坐标绑定定位、锚定位(anchors)、定位器和布局管理器。 说到 anchors,可能很多人都不太了解,它是 QML 中一个非常重要的概念,主要提供了一种相…

C语言几种判断语句简述

C 判断 判断结构要求程序员指定一个或多个要评估或测试的条件&#xff0c;以及条件为真时要执行的语句&#xff08;必需的&#xff09;和条件为假时要执行的语句&#xff08;可选的&#xff09;。 C 语言把任何非零和非空的值假定为 true&#xff0c;把零或 null 假定为 fals…