前言

加更 ... 

测试用例

ajax 发送请求 

<!DOCTYPE html>
<html lang="en">
<head>
  <meta charset="UTF-8">
  <title>01Jquery</title>
</head>
<body>

<span id="resp">

</span>

<script src="../js/jquery.min.js" type="text/javascript"></script>
<script>
  // When data is an object, jQuery generates the data string from the object's key/value pairs unless the processData option is set to false. For example, { a: "bc", d: "e,f" } is converted to the string "a=bc&d=e%2Cf"
  $.ajax({
    url: "/HelloWorld/listForm",
    type: "POST",
    contentType : "application/x-www-form-urlencoded",
    headers : {
      "header01" : "thisIsHeader01",
      "header02" : "thisIsHeader02"
    },
    data : {
      param01 : "thisIsParam01",
      param02 : "thisIsParam02"
    },
    success: function (result) {
      $("#resp").text(JSON.stringify(result))
    }
  });

  // $.ajax({
  //   url: "/HelloWorld/listJson",
  //   type: "POST",
  //   contentType: "application/json",
  //   headers: {
  //     "header01": "thisIsHeader01",
  //     "header02": "thisIsHeader02"
  //   },
  //   data: JSON.stringify({
  //     param01: "thisIsParam01",
  //     param02: "thisIsParam02"
  //   }),
  //   success: function (result) {
  //     $("#resp").text(JSON.stringify(result))
  //   }
  // });
</script>

</body>
</html>

axios 发送请求 

<!DOCTYPE html>
<html lang="en">
<head>
  <meta charset="UTF-8">
  <title>02Axios</title>
</head>
<body>

<span id="resp">

</span>

<script src="../js/axios.min.js" type="text/javascript"></script>
<script>
  // interceptors
  axios.interceptors.request.use(function (config) {
    // 拦截 request, 处理请求
    console.log("do sth before send request, config : ", config)
    return config;
  }, function (error) {
    return Promise.reject(error);
  });

  axios.interceptors.response.use(function (response) {
    // 拦截 response, 处理响应
    console.log("do sth before receive response, response : ", response)
    return response.data;
  }, function (error) {
    return Promise.reject(error);
  });

  // 默认情况下,axios将JavaScript对象序列化为JSON。 要以application / x-www-form-urlencoded格式发送数据,您可以使用以下选项之一。
  const formEncoded = new URLSearchParams()
  formEncoded.append("param01", "thisIsParam01")
  formEncoded.append("param02", "thisIsParam02")
  const rawForms = {
    param01: "thisIsParam01",
    param02: "thisIsParam02"
  }
  axios.request({
    url: "/HelloWorld/listForm",
    method: "post",
    headers: {
      "header01": "thisIsHeader01",
      "header02": "thisIsHeader02",
      'Content-Type': 'application/x-www-form-urlencoded'
    },
    data: formEncoded
  })
    .then(function (fullResp) {
      // let resp = fullResp.data
      let resp = fullResp
      document.getElementById("resp").innerText = JSON.stringify(resp)
    })

  // axios.request({
  //   url: "/HelloWorld/listJson",
  //   method: "post",
  //   headers: {
  //     "header01": "thisIsHeader01",
  //     "header02": "thisIsHeader02",
  //     'Content-Type': 'application/json'
  //   },
  //   data: {
  //     param01: "thisIsParam01",
  //     param02: "thisIsParam02"
  //   }
  // })
  //   .then(function (fullResp) {
  //     //let resp = fullResp.data
  //     let resp = fullResp
  //     document.getElementById("resp").innerText = JSON.stringify(resp)
  //   })

</script>

</body>
</html>

处理以上请求的后端服务, 简单的处理, 获取到请求头 和 参数, 然后 响应给客户端 

通过 @RequesParam[request.getParameter("$paramName")], 或者 @RequestBody 来获取参数 

通过 @RequestHeader 来获取请求头 

/**
 * HelloWorldController
 *
 * @author Jerry.X.He <970655147@qq.com>
 * @version 1.0
 * @date 2022-02-27 21:21
 */
@RestController
@RequestMapping("/HelloWorld")
public class HelloWorldController {

    @RequestMapping("/listForm")
    public List<JSONObject> listForm(
            @RequestHeader("header01") String header01,
            @RequestHeader("header02") String header02,
            String param01,
            String param02
    ) {
        List<JSONObject> result = new ArrayList<>();
        result.add(wrapEntity("header01", header01));
        result.add(wrapEntity("header02", header02));
        result.add(wrapEntity("param01", param01));
        result.add(wrapEntity("param02", param02));
        return result;
    }

    @RequestMapping("/listJson")
    public List<JSONObject> listJson(
            @RequestHeader("header01") String header01,
            @RequestHeader("header02") String header02,
            @RequestBody  JSONObject param01
    ) {
        List<JSONObject> result = new ArrayList<>();
        result.add(wrapEntity("header01", header01));
        result.add(wrapEntity("header02", header02));
        result.add(wrapEntity("param01", JSON.toJSONString(param01)));
        return result;
    }

    // wrapEntity
    public JSONObject wrapEntity(String name, String age) {
        JSONObject result = new JSONObject();
        result.put("name", name);
        result.put("age", age);
        return result;
    }

}

http 请求 和 http 响应

http 请求包含了三部分, 请求行, 请求头, 请求体 

请求行是第一行, 包含了 请求方法, uri, http协议版本 

请求行 和 请求头 通过 换行回车 分割, 接下来的 Host, Connection, 等等属于请求头 

请求体 和 请求头之间通过 两个换行回车分割, "param01=thisIsParam01&param02=thisIsParam02" 属于请求体 

POST /HelloWorld/listForm HTTP/1.1 
Host: localhost
Connection: keep-alive
Content-Length: 43
Pragma: no-cache
Cache-Control: no-cache
sec-ch-ua: " Not A;Brand";v="99", "Chromium";v="99", "Google Chrome";v="99"
header01: thisIsHeader01
sec-ch-ua-mobile: ?0
User-Agent: Mozilla/5.0 (Macintosh; Intel Mac OS X 10_14_6) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/99.0.4844.45 Safari/537.36
Content-Type: application/x-www-form-urlencoded
Accept: */*
X-Requested-With: XMLHttpRequest
sec-ch-ua-platform: "macOS"
header02: thisIsHeader02
Origin: http://localhost
Sec-Fetch-Site: same-origin
Sec-Fetch-Mode: cors
Sec-Fetch-Dest: empty
Referer: http://localhost/httpClient/01Jquery.html
Accept-Encoding: gzip, deflate, br
Accept-Language: en-US,en;q=0.9,zh-CN;q=0.8,zh;q=0.7
Cookie: XXL_JOB_LOGIN_IDENTITY=7b226964223a312c22757365726e616d65223a2261646d696e222c2270617373776f7264223a226531306164633339343962613539616262653536653035376632306638383365222c22726f6c65223a312c227065726d697373696f6e223a6e756c6c7d
 
param01=thisIsParam01¶m02=thisIsParam02

http 响应包含了三部分, 状态行, 响应头, 响应体 

和上面 http 请求的结构类似, 不过状态行 是包含的是 http协议版本 加上 状态码 

HTTP/1.1 200
Server: nginx/1.19.9
Date: Mon, 28 Feb 2022 12:57:10 GMT
Content-Type: application/json;charset=UTF-8
Transfer-Encoding: chunked
Connection: keep-alive
backendIP: 192.168.31.184:8080

b0
[{"name":"header01","age":"thisIsHeader01"},{"name":"header02","age":"thisIsHeader02"},{"name":"param01","age":"{\"param01\":\"thisIsParam01\",\"param02\":\"thisIsParam02\"}"}]
0

form表单 传递参数 

ajax 配置 contentType 为 application/x-www-form-urlencoded, 然后参数直接放到 data 中即可 

默认 jquery 会将参数处理为 表单参数格式

// When data is an object, jQuery generates the data string from the object's key/value pairs unless the processData option is set to false. For example, { a: "bc", d: "e,f" } is converted to the string "a=bc&d=e%2Cf"
  $.ajax({
    url: "/HelloWorld/listForm",
    type: "POST",
    contentType : "application/x-www-form-urlencoded",
    headers : {
      "header01" : "thisIsHeader01",
      "header02" : "thisIsHeader02"
    },
    data : {
      param01 : "thisIsParam01",
      param02 : "thisIsParam02"
    },
    success: function (result) {
      $("#resp").text(JSON.stringify(result))
    }
  });

发送给服务器的 http 请求为 

POST /HelloWorld/listForm HTTP/1.1
Host: localhost
Connection: keep-alive
Content-Length: 43
Pragma: no-cache
Cache-Control: no-cache
sec-ch-ua: " Not A;Brand";v="99", "Chromium";v="99", "Google Chrome";v="99"
header01: thisIsHeader01
sec-ch-ua-mobile: ?0
User-Agent: Mozilla/5.0 (Macintosh; Intel Mac OS X 10_14_6) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/99.0.4844.45 Safari/537.36
Content-Type: application/x-www-form-urlencoded
Accept: */*
X-Requested-With: XMLHttpRequest
sec-ch-ua-platform: "macOS"
header02: thisIsHeader02
Origin: http://localhost
Sec-Fetch-Site: same-origin
Sec-Fetch-Mode: cors
Sec-Fetch-Dest: empty
Referer: http://localhost/httpClient/01Jquery.html
Accept-Encoding: gzip, deflate, br
Accept-Language: en-US,en;q=0.9,zh-CN;q=0.8,zh;q=0.7
Cookie: XXL_JOB_LOGIN_IDENTITY=7b226964223a312c22757365726e616d65223a2261646d696e222c2270617373776f7264223a226531306164633339343962613539616262653536653035376632306638383365222c22726f6c65223a312c227065726d697373696f6e223a6e756c6c7d

param01=thisIsParam01¶m02=thisIsParam02

浏览器中效果如下 

11 ajax 和 axios_ajax

axios 这边默认是序列化为 json, 需要表单格式, 要表单格式的话, 需要手动转换[有 URLSearchParams 相关api]

// 默认情况下,axios将JavaScript对象序列化为JSON。 要以application / x-www-form-urlencoded格式发送数据,您可以使用以下选项之一。
  const formEncoded = new URLSearchParams()
  formEncoded.append("param01", "thisIsParam01")
  formEncoded.append("param02", "thisIsParam02")
  const rawForms = {
    param01: "thisIsParam01",
    param02: "thisIsParam02"
  }
  axios.request({
    url: "/HelloWorld/listForm",
    method: "post",
    headers: {
      "header01": "thisIsHeader01",
      "header02": "thisIsHeader02",
      'Content-Type': 'application/x-www-form-urlencoded'
    },
    data: formEncoded
  })
    .then(function (fullResp) {
      // let resp = fullResp.data
      let resp = fullResp
      document.getElementById("resp").innerText = JSON.stringify(resp)
    })

发送给服务器的 http 请求为 

POST /HelloWorld/listForm HTTP/1.1
Host: localhost
Connection: keep-alive
Content-Length: 43
Pragma: no-cache
Cache-Control: no-cache
sec-ch-ua: " Not A;Brand";v="99", "Chromium";v="99", "Google Chrome";v="99"
header01: thisIsHeader01
sec-ch-ua-mobile: ?0
User-Agent: Mozilla/5.0 (Macintosh; Intel Mac OS X 10_14_6) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/99.0.4844.45 Safari/537.36
Content-Type: application/x-www-form-urlencoded
Accept: application/json, text/plain, */*
sec-ch-ua-platform: "macOS"
header02: thisIsHeader02
Origin: http://localhost
Sec-Fetch-Site: same-origin
Sec-Fetch-Mode: cors
Sec-Fetch-Dest: empty
Referer: http://localhost/httpClient/02Axios.html
Accept-Encoding: gzip, deflate, br
Accept-Language: en-US,en;q=0.9,zh-CN;q=0.8,zh;q=0.7
Cookie: XXL_JOB_LOGIN_IDENTITY=7b226964223a312c22757365726e616d65223a2261646d696e222c2270617373776f7264223a226531306164633339343962613539616262653536653035376632306638383365222c22726f6c65223a312c227065726d697373696f6e223a6e756c6c7d

param01=thisIsParam01¶m02=thisIsParam02

浏览器中效果如下 

11 ajax 和 axios_ios_02

application/json 传递参数 

ajax 配置 contentType 为 application/json,  默认 jquery 会将参数处理为 表单参数格式, 所以需要将参数手动处理为 json 字符串, 这里是基于 JSON.stringify 

$.ajax({
    url: "/HelloWorld/listJson",
    type: "POST",
    contentType: "application/json",
    headers: {
      "header01": "thisIsHeader01",
      "header02": "thisIsHeader02"
    },
    data: JSON.stringify({
      param01: "thisIsParam01",
      param02: "thisIsParam02"
    }),
    success: function (result) {
      $("#resp").text(JSON.stringify(result))
    }
  });

发送给服务器的 http 请求为 

POST /HelloWorld/listJson HTTP/1.1
Host: localhost
Connection: keep-alive
Content-Length: 53
Pragma: no-cache
Cache-Control: no-cache
sec-ch-ua: " Not A;Brand";v="99", "Chromium";v="99", "Google Chrome";v="99"
header01: thisIsHeader01
sec-ch-ua-mobile: ?0
User-Agent: Mozilla/5.0 (Macintosh; Intel Mac OS X 10_14_6) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/99.0.4844.45 Safari/537.36
Content-Type: application/json
Accept: */*
X-Requested-With: XMLHttpRequest
sec-ch-ua-platform: "macOS"
header02: thisIsHeader02
Origin: http://localhost
Sec-Fetch-Site: same-origin
Sec-Fetch-Mode: cors
Sec-Fetch-Dest: empty
Referer: http://localhost/httpClient/01Jquery.html
Accept-Encoding: gzip, deflate, br
Accept-Language: en-US,en;q=0.9,zh-CN;q=0.8,zh;q=0.7
Cookie: XXL_JOB_LOGIN_IDENTITY=7b226964223a312c22757365726e616d65223a2261646d696e222c2270617373776f7264223a226531306164633339343962613539616262653536653035376632306638383365222c22726f6c65223a312c227065726d697373696f6e223a6e756c6c7d

{"param01":"thisIsParam01","param02":"thisIsParam02"}

浏览器中效果如下 

11 ajax 和 axios_http_03

axios 这边默认是序列化为 json, 不用做任何转化 

axios.request({
    url: "/HelloWorld/listJson",
    method: "post",
    headers: {
      "header01": "thisIsHeader01",
      "header02": "thisIsHeader02",
      'Content-Type': 'application/json'
    },
    data: {
      param01: "thisIsParam01",
      param02: "thisIsParam02"
    }
  })
    .then(function (fullResp) {
      let resp = fullResp.data
      // let resp = fullResp
      document.getElementById("resp").innerText = JSON.stringify(resp)
    })

发送给服务器的 http 请求为 

POST /HelloWorld/listJson HTTP/1.1
Host: localhost
Connection: keep-alive
Content-Length: 53
Pragma: no-cache
Cache-Control: no-cache
sec-ch-ua: " Not A;Brand";v="99", "Chromium";v="99", "Google Chrome";v="99"
header01: thisIsHeader01
sec-ch-ua-mobile: ?0
User-Agent: Mozilla/5.0 (Macintosh; Intel Mac OS X 10_14_6) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/99.0.4844.45 Safari/537.36
Content-Type: application/json
Accept: application/json, text/plain, */*
sec-ch-ua-platform: "macOS"
header02: thisIsHeader02
Origin: http://localhost
Sec-Fetch-Site: same-origin
Sec-Fetch-Mode: cors
Sec-Fetch-Dest: empty
Referer: http://localhost/httpClient/02Axios.html
Accept-Encoding: gzip, deflate, br
Accept-Language: en-US,en;q=0.9,zh-CN;q=0.8,zh;q=0.7
Cookie: XXL_JOB_LOGIN_IDENTITY=7b226964223a312c22757365726e616d65223a2261646d696e222c2270617373776f7264223a226531306164633339343962613539616262653536653035376632306638383365222c22726f6c65223a312c227065726d697373696f6e223a6e756c6c7d

{"param01":"thisIsParam01","param02":"thisIsParam02"}

浏览器中效果如下 

11 ajax 和 axios_http_04

axios 的拦截器 

呵呵 常规的拦截处理, 在发送请求之前对于请求的统一转换处理, 以及在收到响应之后, 对于响应的统一转换处理 

这里 response 拦截处理, 直接获取了 reponse.data, 那么业务这边拿到的就是 reponse.data 

// interceptors
  axios.interceptors.request.use(function (config) {
    // 拦截 request, 处理请求
    console.log("do sth before send request, config : ", config)
    return config;
  }, function (error) {
    return Promise.reject(error);
  });

  axios.interceptors.response.use(function (response) {
    // 拦截 response, 处理响应
    console.log("do sth before receive response, response : ", response)
    return response.data;
  }, function (error) {
    return Promise.reject(error);
  });

下面的 then, 回调处理 拿到的 fullResp 即为服务端返回的响应数据, 而不是 axios 本身封装的一层 响应实体了 

可以和上面没有加 axios 拦截器的场景进行参照对比一下 

axios.request({
    url: "/HelloWorld/listJson",
    method: "post",
    headers: {
      "header01": "thisIsHeader01",
      "header02": "thisIsHeader02",
      'Content-Type': 'application/json'
    },
    data: {
      param01: "thisIsParam01",
      param02: "thisIsParam02"
    }
  })
    .then(function (fullResp) {
      // let resp = fullResp.data
      let resp = fullResp
      document.getElementById("resp").innerText = JSON.stringify(resp)
    })

整个网关流程梳理

本测试用例基于 nginx 来提供前端服务, 以及代理后端服务 

nginx 服务器的 docker-compose 如下, 配置了容器中的 root目录 挂载在 宿主机的 "/Users/jerry/WebstormProjects/HelloWorld", 然后 "http://localhost/httpClient/01Jquery.html" 访问的就是 "/Users/jerry/WebstormProjects/HelloWorld/httpClient/01Jquery.html" 

version: "2"
  
services:
  nginx:
    container_name: nginx
    image: nginx:latest
    ports:
      - "80:80"
    volumes:
      - ./data:/etc/nginx
#      - ./html:/usr/share/nginx/html
      - /Users/jerry/WebstormProjects/HelloWorld:/usr/share/nginx/html

项目目录结构如下

11 ajax 和 axios_client_05

nginx 配置文件如下 

root目录 对应的是容器中的 /usr/share/nginx/html, 对应于挂载在宿主机的 /Users/jerry/WebstormProjects/HelloWorld 

配置了 "/HelloWorld" 打头的服务, 转发到 192.168.31.184:8080, 也就是我们的后端服务 

上面测试的 "/HelloWorld/listForm", "/HelloWorld/listJson" 都是会转发到 192.168.31.184:8080 

server {
    listen       80;
    listen  [::]:80;
    server_name  localhost;

    #charset koi8-r;
    #access_log  /var/log/nginx/host.access.log  main;

    location / {
        root   /usr/share/nginx/html;
        index  index.html index.htm;
    }

    location ~ /HelloWorld {
        proxy_pass   http://192.168.31.184:8080;
       add_header backendIP $upstream_addr;
       proxy_set_header SSL-Client-Cert $ssl_client_cert;
       proxy_set_header name jerry;
    }

    #error_page  404              /404.html;

    # redirect server error pages to the static page /50x.html
    #
    error_page   500 502 503 504  /50x.html;
    location = /50x.html {
        root   /usr/share/nginx/html;
    }

}