• Node.js 使用 Express 写接口、CORS 跨域资源共享


    使用 Express 写接口

    1.创建基本的服务器

    在这里插入图片描述

    2.创建 API 路由模块

    在这里插入图片描述

    3.编写 GET 接口

    在这里插入图片描述

    4.编写 POST 接口

    在这里插入图片描述
    注意:如果要获取 URL-encoded 格式的请求体数据,必须配置中间件 app.use(express.urlencoded({ extended: false }))

    CORS 跨域资源共享解决方法

    刚才编写的 GET 和 POST接口,存在一个很严重的问题:不支持跨域请求
    解决接口跨域问题的方案主要有两种:
    ① CORS(主流的解决方案,推荐使用)
    ② JSONP(有缺陷的解决方案:只支持 GET 请求)

    cors 是 Express 的一个第三方中间件。通过安装和配置 cors 中间件,可以很方便地解决跨域问题。
    使用步骤分为如下 3 步:

    • ① 运行 npm install cors 安装中间件
    • ② 使用 const cors = require(‘cors’) 导入中间件
    • ③ 在路由之前调用 app.use(cors()) 配置中间件

    CORS 的注意事项

    • ① CORS 主要在服务器端进行配置。客户端浏览器无须做任何额外的配置,即可请求开启了 CORS 的接口。
    • ② CORS 在浏览器中有兼容性。只有支持 XMLHttpRequest Level2 的浏览器,才能正常访问开启了 CORS 的服
      务端接口(例如:IE10+、Chrome4+、FireFox3.5+)。
    // 1、引入express
    const express = require('express');
    
    // 2、创建应用对象
    const app = express();
    
    // 3、创建路由规则
    // request 是对请求报文的封装
    // response 是对相应报文的封装
    app.get("/cors-server",(request,response)=>{
        response.setHeader("Access-Control-Allow-Origin","*");
        response.setHeader("Access-Control-Allow-Headers","*");
        response.setHeader("Access-Control-Allow-Methods","*");
        response.send("hello CORS");
    });
    
    // 4、监听端口启动服务
    app.listen(8000,()=>{
        console.log("服务已经启动,8000端口监听中。。。");
    })
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20

    CORS 响应头部 - Access-Control-Allow-Origin

    如果指定了 Access-Control-Allow-Origin 字段的值为通配符 *,表示允许来自任何域的请求,示例代码如下:
    在这里插入图片描述

    CORS 响应头部 - Access-Control-Allow-Headers

    默认情况下,CORS 仅支持客户端向服务器发送如下的 9 个请求头:
    Accept、Accept-Language、Content-Language、DPR、Downlink、Save-Data、Viewport-Width、Width 、Content-Type (值仅限于 text/plain、multipart/form-data、application/x-www-form-urlencoded 三者之一)
    如果客户端向服务器发送了额外的请求头信息,则需要在服务器端,通过 Access-Control-Allow-Headers 对额外的请求头进行声明,否则这次请求会失败!
    在这里插入图片描述

    CORS 响应头部 - Access-Control-Allow-Methods

    默认情况下,CORS 仅支持客户端发起 GET、POST、HEAD 请求。
    如果客户端希望通过 PUT、DELETE 等方式请求服务器的资源,则需要在服务器端,通过 Access-Control-Alow-Methods来指明实际请求所允许使用的 HTTP 方法。
    示例代码如下:
    在这里插入图片描述

    6.JSONP

    概念:浏览器端通过 < script> 标签的 src 属性,请求服务器上的数据,同时,服务器返回一个函数的调用。这种请求数据的方式叫做 JSONP。

    特点:

    • ① JSONP 不属于真正的 Ajax 请求,因为它没有使用 XMLHttpRequest 这个对象。
    • ② JSONP 仅支持 GET 请求,不支持 POST、PUT、DELETE 等请求。

    如果项目中已经配置了 CORS 跨域资源共享,为了防止冲突,必须在配置 CORS 中间件之前声明 JSONP 的接口。否则JSONP 接口会被处理成开启了 CORS 的接口。示例代码如下:
    在这里插入图片描述

    实现 JSONP 接口的步骤

    • ① 获取客户端发送过来的回调函数的名字
    • ② 得到要通过 JSONP 形式发送给客户端的数据
    • ③ 根据前两步得到的数据,拼接出一个函数调用的字符串
    • ④ 把上一步拼接得到的字符串,响应给客户端的

    在网页中使用 jQuery 发起 JSONP 请求

    调用 $.ajax() 函数,提供 JSONP 的配置选项,从而发起 JSONP 请求,示例代码如下:
    在这里插入图片描述

    附录:源码

    简单接口

    var express = require('express');//引入express模块
    var app = express();
    
    //响应接口
    app.get('/api/get', function (req, res) {
        //数据
        let result = {
            "code": "200",
            "data": {
                "msg": "登陆成功",
                "success": "true",
                "token": "admin_token",
                "username": "admin"
            }
        }
        res.send(result)
    })
    
    app.post('/api/post', (req, res) => {
        // 调用 express 提供的 res.send() 方法,向客户端响应一个 文本字符串
        let result = {
            "code": "200",
            "data": {
                "msg": "登陆成功",
                "success": "true",
                "token": "admin_token",
                "username": "admin"
            }
        }
        res.send(result)
    });
    
    //定义端口,此处所用为3000端口,可自行更改
    var server = app.listen(3000, function () {
        console.log('runing http://localhost:3000/');
    })
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37

    分模块 jsonp跨域原理

    // 导入 express
    const express = require('express')
    // 创建服务器实例
    const app = express()
    
    // 配置解析表单数据的中间件
    app.use(express.urlencoded({ extended: false }))
    
    //------------------------------------------------
    //JSONP 接口具体的实现过程
    // 必须在配置 cors 中间件之前,配置 JSONP 的接口
    app.get('/api/jsonp', (req, res) => {
      // TODO: 定义 JSONP 接口具体的实现过程
      // 1. 得到函数的名称
      const funcName = req.query.callback
      // 2. 定义要发送到客户端的数据对象
      const data = { name: 'zs', age: 22 }
      // 3. 拼接出一个函数的调用
      const scriptStr = `${funcName}(${JSON.stringify(data)})`
      // 4. 把拼接的字符串,响应给客户端
      res.send(scriptStr)
    })
    ---------------------------------------------------
    //使用cors解决跨域
    // 一定要在路由之前,配置 cors 这个中间件,从而解决接口跨域的问题
    const cors = require('cors')
    app.use(cors())
    
    // 导入路由模块
    const router = require('./apiRouter')
    // 把路由模块,注册到 app 上
    app.use('/api', router)
    
    // 启动服务器
    app.listen(80, () => {
      console.log('express server running at http://127.0.0.1')
    })
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37

    apiRouter.js

    const express = require('express')
    const router = express.Router()
    
    // 在这里挂载对应的路由
    router.get('/get', (req, res) => {
      // 通过 req.query 获取客户端通过查询字符串,发送到服务器的数据
      const query = req.query
      // 调用 res.send() 方法,向客户端响应处理的结果
      res.send({
        status: 0, // 0 表示处理成功,1 表示处理失败
        msg: 'GET 请求成功!', // 状态的描述
        data: query, // 需要响应给客户端的数据
      })
    })
    
    // 定义 POST 接口
    router.post('/post', (req, res) => {
      // 通过 req.body 获取请求体中包含的 url-encoded 格式的数据
      const body = req.body
      // 调用 res.send() 方法,向客户端响应结果
      res.send({
        status: 0,
        msg: 'POST 请求成功!',
        data: body,
      })
    })
    
    // 定义 DELETE 接口
    router.delete('/delete', (req, res) => {
      res.send({
        status: 0,
        msg: 'DELETE请求成功',
      })
    })
    
    module.exports = router
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
  • 相关阅读:
    基于Springboot+Mybatis+mysql+vue技术交流博客论坛系统
    Leetcode143. 重排链表
    el-table中保留分页选中
    $attrs,$listeners
    数据湖&湖仓一体简介
    vscode自动升级后无法打开远程解决方案
    Linux计划任务at和cron命令的使用
    Karl Guttag:AR眼镜应根据用途来设计,VST并未解决技术难题
    Python:自动化处理PDF文档集合,提取文献标题、合并文献PDF并生成目录和页码
    Docker笔记-09 Docker Compose
  • 原文地址:https://blog.csdn.net/qq_43472877/article/details/127674348