初学相关工具

express框架

npm init –yes

npm i express

// 引入express
const { request, response } = require("express");
const express =require("express");

// 创建应用对象
const app = express();

// 创建路由规则
// request是对请求报文的封装
// response是对响应报文的封装
app.get(或者post)('/',(request,response)=>{
    response.setHeader('Access-Control-Allow-Origin','*');   //重要
    response.send("huadhf")
}); 

// 监听端口启动服务
app.listen(8088, ()=>{
    console.log("服务已启动,8088端口监听中……");
});

nodemon 自动刷新 :

一次安装,终身受用。

安装:npm install -g nodemon
使用方法:nodemon [your node app]

XMLHttpRequest

//创建对象
const xhr = new XMLHttpRequest();
//初始化  设置请求方法和url
xhr.open('GET','HTTP://127.0.0.1:8000');
//发送
xhr.send();
//事件绑定:
//readystate是xhr对象中的属性0(未初始化),1(open完),2(send完),3(服务端返回部分接口),4(服务端返回所有接口),change是改变的意思。
xhr.onreadystatechang = function (){
//判断(服务端返回所有结果)
if(xhr.readystate ==4){
//判断响应码  200  404  403 401  500
if(xhr.status >=200&&xhr.status<=300){
//处理结果 行 头 空行 体
console.log(xhr.status);//状态码
console.log(xhr.statusText);//状态字符串
console.log(xhr.getALLResponseHeaders());//所有响应头
console.log(xhr.response);//响应体
    xxxxx.innerHTML=xhr.response;
}
}



};

post同理 方法改为post即可;

设置请求体:

get:

//初始化  设置请求方法和url
xhr.open('GET','HTTP://127.0.0.1:8000?a=100&b=300');


在url后加’?‘然后传参。

post:

//发送
xhr.send('a=110&b=120');

//发送
xhr.send('a:1231&b:1231');

//发送
xhr.send('1231fauyghrfu');   什么都行,靠后端解析,尽量规范 

设置请求头:

前面用于请求体内容类型,后面是参数查询字符串的类型(固定写法)前后端都能写

一般放信息认证的内容。

//初始化  设置请求方法和url
xhr.open('GET','HTTP://127.0.0.1:8000');

xhr.setRequestHeader('Content-Type','application/x-www-form-urlencoded');(预定义 )  前端:设置那些请求头内容
response.setHeader('Access-Control-Allow-Origin','*');(预定义)后端:*代表什么都能接受
有很多种预定义,百度。
xhr.setRequestHeader('xxxxx','xxxxx');(自定义)

服务端响应json数据

后端示例:
app.all('/prpr-json',(request,response)=>{
    // 设置响应头 允许跨域
    response.setHeader('Access-Control-Allow-Origin','*');
    // 响应头
    response.setHeader('Access-Control-Allow-Headers','*');
    // 一个数据
    const data = {
        name:"NanIvy"
    };
    // 对对象进行转换
    let str =JSON.stringify(data);
    response.send(str);

}); 


前端接受处理转换:   (手动转换)

            const xhr = new XMLHttpRequest();

            xhr.open('GET','http://127.0.0.1:8088/prpr-json');

            xhr.send();

            xhr.onreadystatechange =()=>{
                if(xhr.readyState===4){
                    if(xhr.status>=200&xhr.status<=300){
                        let data = JSON.parse(xhr.response);
                        dd.innerHTML = data.name;
                    }
                }
            }


            (主动转换)

             window.onkeydown =()=>{
            const xhr = new XMLHttpRequest();

            xhr.responseType='json';//*******************************************

            xhr.open('GET','http://127.0.0.1:8088/prpr-json');

            xhr.send();

            xhr.onreadystatechange =()=>{
                if(xhr.readyState===4){
                    if(xhr.status>=200&xhr.status<=300){
                        // let data = JSON.parse(xhr.response);    手动转换
                        // dd.innerHTML = data.name;
                        dd.innerHTML = xhr.response.name;//**************************
                    }
                }
            }

请求超时

多了三个xhr操作

        const dd = document.getElementById('es');
        window.onkeydown =()=>{
            const xhr = new XMLHttpRequest();   

            xhr.open('GET','http://127.0.0.1:8088/timeout');

            xhr.send();
            // 设置超时2s*****************************
            xhr.timeout = 2000;
            // 超时回调*******************************
            xhr.ontimeout  = ()=>{
                alert("网络异常,请稍后重试");
            };
            // 报错监视*******************************
            xhr.onerror = ()=>{
                alert("(网络)错误");
            }
            xhr.onreadystatechange =()=>{
                if(xhr.readyState===4){
                    if(xhr.status>=200&xhr.status<=300){
                        dd.innerHTML = xhr.response;
                    }
                }
            }
        }




// 后端:定时器模拟超时:请求超时
app.get('/timeout',(request,response)=>{
    response.setHeader('Access-Control-Allow-Origin','*');
    setTimeout(() => {
        response.send("延时响应");
    }, 3000);
}); 



取消发送

 var p = document.getElementsByTagName("button")[0];
        var c = document.getElementsByTagName("button")[1];
        var tt =document.getElementsByClassName('s')[0];

        let xhr = null ;       //全局************************
        p.onclick = function(){
            xhr = new XMLHttpRequest()
            xhr.open('GET','http://127.0.0.1:8088/timeout');
            xhr.send();

        }
        // 取消按钮******************************************
        c.onclick = function(){
           xhr.abort();
         }

         // 后端
app.get('/timeout',(request,response)=>{
    response.setHeader('Access-Control-Allow-Origin','*');
    setTimeout(() => {
        response.send("延时响应");
    }, 3000);
}); 

重复请求问题

let xhr = null ;    
        //设定标识********************************
        let isSending = false;

        p.onclick = function(){
            if(isSending) xhr.abort();      //如果是在发送,取消上一个发送,接下来开始新发送
            isSending = true;             //开始发送,标识改为true
            xhr = new XMLHttpRequest()
            xhr.open('GET','http://127.0.0.1:8088/timeout');
            xhr.send();
            if(xhr.readyState===4){
                isSending=false;          //完成,修改标识变量
            }
        }


        app.get('/timeout',(request,response)=>{
    response.setHeader('Access-Control-Allow-Origin','*');
    setTimeout(() => {
        response.send("延时响应");
    }, 3000);
}); 

同源策略

ajax默认遵循同源策略:

同源(同一个来源):协议(http),域名(xx.xx),端口号必须完全相同。

违背同源策略就是跨域。

解决跨域

为什么跨域?——服务器的能力是有上限哒

CORS:

有多个响应头字段:

Access-Control-Allow-Origin
指示请求的资源能共享给哪些域。
Access-Control-Allow-Credentials
指示当请求的凭证标记为 true 时,是否响应该请求。
Access-Control-Allow-Headers
用在对预请求的响应中,指示实际的请求中可以使用哪些 HTTP 头。
Access-Control-Allow-Methods
指定对预请求的响应中,哪些 HTTP 方法允许访问请求的资源。
Access-Control-Expose-Headers
指示哪些 HTTP 头的名称能在响应中列出。
Access-Control-Max-Age
指示预请求的结果能被缓存多久。
Access-Control-Request-Headers
用于发起一个预请求,告知服务器正式请求会使用那些 HTTP 头。
Access-Control-Request-Method
用于发起一个预请求,告知服务器正式请求会使用哪一种 HTTP 请求方法。
Origin
指示获取资源的请求是从什么域发起的。

例子:

// 设置响应头 允许跨域
response.setHeader('Access-Control-Allow-Origin','*');   所有
response.setHeader('Access-Control-Allow-Origin','HTTP://127.0.0.1:5500');   只有该端口网页可发送请求


//请求方法
response.setHeader('Access-Control-Allow-Methods',''*');允许所有请求方法 如put,delete,conect 等等

// 响应头
response.setHeader('Access-Control-Allow-Headers','*');
//允许各种自定义头

axios

axios的引入:

Using npm:

$ npm install axios

Using bower:

$ bower install axios

Using yarn:

$ yarn add axios

Using jsDelivr CDN:

<script src="https://cdn.jsdelivr.net/npm/axios/dist/axios.min.js"></script>

Using unpkg CDN:

<script src="https://unpkg.com/axios/dist/axios.min.js"></script>

axios初体验

增删改查: 使用基本操作

<!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>
    <script src="https://unpkg.com/axios/dist/axios.min.js"></script>
</head>
<body>
    <button>get 请求</button>
    <button>post 请求</button>
    <button>put 请求</button>
    <button>delete 请求</button>
    <script>
        const oget=document.getElementsByTagName('button')[0];
        oget.onclick = function(){
            axios({
                method:'GET',
                url:'http://127.0.0.1:3000/posts/2',
            }).then(response =>{
                console.log(response);
            });
        }

        const opost=document.getElementsByTagName('button')[1];
        opost.onclick = function(){
            axios({
                method:'POST',
                url:'http://127.0.0.1:3000/posts',
                // 设置请求体
                data:{
                    title:'post-test',
                    author:'nico'
                }
            }).then(response =>{
                console.log(response);
            });
        }

        const oput=document.getElementsByTagName('button')[2];
        oput.onclick = function(){
            axios({
                method:'PUT',
                url:'http://127.0.0.1:3000/posts/3',
                // 设置请求体
                data:{
                    title:'put-test',
                    author:'nanivy'
                }
            }).then(response =>{
                console.log(response);
            });
        }

        const odelete=document.getElementsByTagName('button')[3];
        odelete.onclick = function(){
            axios({
                method:'delete',
                url:'http://127.0.0.1:3000/posts/3',
                // 设置请求体

            }).then(response =>{
                console.log(response);
            });
        }

    </script>
</body>
</html>

axios请求方法

        方式一:  axios.post('/url',data,config)

      方式二:  axios({
        method: 'post',
        url: '/url',
        data: data,
        config: config
      })

其他方法同理。

参考:http://www.axios-js.com/zh-cn/docs/

请求方法的别名

为方便起见,为所有支持的请求方法提供了别名

axios.request(config)

axios.get(url[, config])

axios.delete(url[, config])

axios.head(url[, config])

axios.options(url[, config])

axios.post(url[, data[, config]])

axios.put(url[, data[, config]])

axios.patch(url[, data[, config]])

注意

在使用别名方法时, urlmethoddata 这些属性都不必在配置中指定。

请求配置

这些是创建请求时可以用的配置选项。只有 url 是必需的。如果没有指定 method,请求将默认使用 get方法。

{
   // `url` 是用于请求的服务器 URL
  url: '/user',

  // `method` 是创建请求时使用的方法
  method: 'get', // default

  // `baseURL` 将自动加在 `url` 前面,除非 `url` 是一个绝对 URL。
  // 它可以通过设置一个 `baseURL` 便于为 axios 实例的方法传递相对 URL
  baseURL: 'https://some-domain.com/api/',

  // `transformRequest` 允许在向服务器发送前,修改请求数据
  // 只能用在 'PUT', 'POST' 和 'PATCH' 这几个请求方法
  // 后面数组中的函数必须返回一个字符串,或 ArrayBuffer,或 Stream
  transformRequest: [function (data, headers) {
    // 对 data 进行任意转换处理
    return data;
  }],

  // `transformResponse` 在传递给 then/catch 前,允许修改响应数据
  transformResponse: [function (data) {
    // 对 data 进行任意转换处理
    return data;
  }],

  // `headers` 是即将被发送的自定义请求头
  headers: {'X-Requested-With': 'XMLHttpRequest'},

  // `params` 是即将与请求一起发送的 URL 参数
  // 必须是一个无格式对象(plain object)或 URLSearchParams 对象
  params: {
    ID: 12345
  },

   // `paramsSerializer` 是一个负责 `params` 序列化的函数
  // (e.g. https://www.npmjs.com/package/qs, http://api.jquery.com/jquery.param/)
  paramsSerializer: function(params) {
    return Qs.stringify(params, {arrayFormat: 'brackets'})
  },

  // `data` 是作为请求主体被发送的数据
  // 只适用于这些请求方法 'PUT', 'POST', 和 'PATCH'
  // 在没有设置 `transformRequest` 时,必须是以下类型之一:
  // - string, plain object, ArrayBuffer, ArrayBufferView, URLSearchParams
  // - 浏览器专属:FormData, File, Blob
  // - Node 专属: Stream
  data: {
    firstName: 'Fred'
  },

  // `timeout` 指定请求超时的毫秒数(0 表示无超时时间)
  // 如果请求话费了超过 `timeout` 的时间,请求将被中断
  timeout: 1000,

   // `withCredentials` 表示跨域请求时是否需要使用凭证(cookie)
  withCredentials: false, // default

  // `adapter` 允许自定义处理请求,以使测试更轻松
  // 返回一个 promise 并应用一个有效的响应 (查阅 [response docs](#response-api)).
  adapter: function (config) {
    /* ... */
  },

 // `auth` 表示应该使用 HTTP 基础验证,并提供凭据
  // 这将设置一个 `Authorization` 头,覆写掉现有的任意使用 `headers` 设置的自定义 `Authorization`头
  auth: {
    username: 'janedoe',
    password: 's00pers3cret'
  },

   // `responseType` 表示服务器响应的数据类型,可以是 'arraybuffer', 'blob', 'document', 'json', 'text', 'stream'
  responseType: 'json', // default

  // `responseEncoding` indicates encoding to use for decoding responses
  // Note: Ignored for `responseType` of 'stream' or client-side requests
  responseEncoding: 'utf8', // default

   // `xsrfCookieName` 是用作 xsrf token 的值的cookie的名称
  xsrfCookieName: 'XSRF-TOKEN', // default

  // `xsrfHeaderName` is the name of the http header that carries the xsrf token value
  xsrfHeaderName: 'X-XSRF-TOKEN', // default

   // `onUploadProgress` 允许为上传处理进度事件
  onUploadProgress: function (progressEvent) {
    // Do whatever you want with the native progress event
  },

  // `onDownloadProgress` 允许为下载处理进度事件
  onDownloadProgress: function (progressEvent) {
    // 对原生进度事件的处理
  },

   // `maxContentLength` 定义允许的响应内容的最大尺寸
  maxContentLength: 2000,

  // `validateStatus` 定义对于给定的HTTP 响应状态码是 resolve 或 reject  promise 。如果 `validateStatus` 返回 `true` (或者设置为 `null` 或 `undefined`),promise 将被 resolve; 否则,promise 将被 rejecte
  validateStatus: function (status) {
    return status >= 200 && status < 300; // default
  },

  // `maxRedirects` 定义在 node.js 中 follow 的最大重定向数目
  // 如果设置为0,将不会 follow 任何重定向
  maxRedirects: 5, // default

  // `socketPath` defines a UNIX Socket to be used in node.js.
  // e.g. '/var/run/docker.sock' to send requests to the docker daemon.
  // Only either `socketPath` or `proxy` can be specified.
  // If both are specified, `socketPath` is used.
  socketPath: null, // default

  // `httpAgent` 和 `httpsAgent` 分别在 node.js 中用于定义在执行 http 和 https 时使用的自定义代理。允许像这样配置选项:
  // `keepAlive` 默认没有启用
  httpAgent: new http.Agent({ keepAlive: true }),
  httpsAgent: new https.Agent({ keepAlive: true }),

  // 'proxy' 定义代理服务器的主机名称和端口
  // `auth` 表示 HTTP 基础验证应当用于连接代理,并提供凭据
  // 这将会设置一个 `Proxy-Authorization` 头,覆写掉已有的通过使用 `header` 设置的自定义 `Proxy-Authorization` 头。
  proxy: {
    host: '127.0.0.1',
    port: 9000,
    auth: {
      username: 'mikeymike',
      password: 'rapunz3l'
    }
  },

  // `cancelToken` 指定用于取消请求的 cancel token
  // (查看后面的 Cancellation 这节了解更多)
  cancelToken: new CancelToken(function (cancel) {
  })
}

tip:data与params

params是添加到url的请求字符串中的,一般用于get请求。
data是添加到请求体(body)中的, 一般用于post请求。

响应结构

某个请求的响应包含以下信息

{
  // `data` 由服务器提供的响应
  data: {},

  // `status` 来自服务器响应的 HTTP 状态码
  status: 200,

  // `statusText` 来自服务器响应的 HTTP 状态信息
  statusText: 'OK',

  // `headers` 服务器响应的头
  headers: {},

   // `config` 是为请求提供的配置信息
  config: {},
 // 'request'
  // `request` is the request that generated this response
  // It is the last ClientRequest instance in node.js (in redirects)
  // and an XMLHttpRequest instance the browser
  request: {}
}

使用 then 时,你将接收下面这样的响应 :

axios.get('/user/12345')
  .then(function(response) {
    console.log(response.data);
    console.log(response.status);
    console.log(response.statusText);
    console.log(response.headers);
    console.log(response.config);
  });

axios配置默认参数:

请求对象中的配置 都是可以设置的:

如:

axios.defaults.method = 'GET';
axios.defaults.baseURL = 'http://localhost:3000'

axios创建实例对象

可以使用自定义配置新建一个 axios 实例

axios.create([config])
const instance = axios.create({
  baseURL: 'https://some-domain.com/api/',
  timeout: 1000,
  headers: {'X-Custom-Header': 'foobar'}
});


instance();
instance({url:‘/xx’}).then(response=>{XXXXX});
instance.get(‘/xx’).then(response=>{XXXXX});

axios拦截器

在请求或响应被 thencatch 处理前拦截它们。

// 添加请求拦截器
axios.interceptors.request.use(function (config) {
    // 在发送请求之前做些什么
    return config;
  }, function (error) {
    // 对请求错误做些什么
    return Promise.reject(error);
  });

// 添加响应拦截器
axios.interceptors.response.use(function (response) {
    // 对响应数据做点什么
    return response;
  }, function (error) {
    // 对响应错误做点什么
    return Promise.reject(error);
  });

稍后移除拦截器,可以这样:

const myInterceptor = axios.interceptors.request.use(function () {/*...*/});
axios.interceptors.request.eject(myInterceptor);

可以为自定义 axios 实例添加拦截器

const instance = axios.create();
instance.interceptors.request.use(function () {/*...*/});

axios取消请求

更偏向使用第二种方法:

使用 CancelToken.source 工厂方法创建 cancel token,像这样:

const CancelToken = axios.CancelToken;
const source = CancelToken.source();

axios.get('/user/12345', {
  cancelToken: source.token
}).catch(function(thrown) {
  if (axios.isCancel(thrown)) {
    console.log('Request canceled', thrown.message);
  } else {
     // 处理错误
  }
});

axios.post('/user/12345', {
  name: 'new name'
}, {
  cancelToken: source.token
})

// 取消请求(message 参数是可选的)
source.cancel('Operation canceled by the user.');

还可以通过传递一个 executor 函数到 CancelToken 的构造函数来创建 cancel token:

const CancelToken = axios.CancelToken;
let cancel;

axios.get('/user/12345', {
  cancelToken: new axios.CancelToken(function(c) {
    cancel = c;
  })
});

// cancel the request
cancel();

防抖示例:

const oget=document.getElementsByTagName('button')[0];
        let cancel=null;
        oget.onclick = function(){
            if(cancel!==null){
                cancel();
            }
            axios({
                method:'GET',
                url:'http://127.0.0.1:3000/posts/2',

                cancelToken: new axios.CancelToken((c)=>{
                    cancel=c;
                })

            }).then(response =>{
                console.log(response);
                cancel=null;
            });
        }

源码待学习


菜菜子新人