十三:对AJAX的理解

114 阅读3分钟

进入前端,听到的第一个专业名词,ajax,也是阿贾克斯,很容易联想到荷甲的阿贾克斯俱乐部。

AJAX是Asynchronous JavaScript and XML 的缩写,指的是通过JavaScript的异步通信,从服务器获取XML文档,从中提取数据,再更新当前页面的对应部分,而不用刷新整个页面。
创建AJAX请求的步骤:

  1. 创建一个XMLHttpRequest对象。
  2. 在这个对象上使用open方法创建一个http请求,open方法所需要的参数是请求的方法,请求的地址、是否异步和用户的认证信息。
  3. 在发起请求前,可以为这个对象添加一些信息和监听函数。比如说可以通过setRequestHeader方法来为请求添加头信息。还可以为这个对象添加一个状态监听函数。
  4. 一个XMLHttpRequest对象一共有5个状态,当他的状态变化时会触发onreadystatechange事件,可以通过设置监听函数,来处理请求成功后的结果。
  5. 当对象的readyState变为4的时候,代表服务器返回的数据接收完成,这个时候可以通过判断请求的状态,如果状态是2xx或者304的话,则代表返回正常。
  6. 这个时候就可以通过response中的数据来对页面进行更新了。
  7. 当对象的属性和监听函数设置完成后,最后调用sent方法来向服务器发起请求,可以传入参数作为发送的数据体。

对于axios,请求拦截器和响应拦截器分别做了哪些工作?

  • 请求拦截器在请求发送前进行必要操作处理,例如添加统一 cookie、请求体加验证、设置请求头等,相当于是对每个接口相同操作的一个封装。
service.interceptors.request.use(config => {

  // 是否需要设置 token

  const isToken = (config.headers || {}).isToken === false

  // 是否需要防止数据重复提交

  const isRepeatSubmit = (config.headers || {}).repeatSubmit === false

  if (getToken() && !isToken) {

    config.headers['Authorization'] = 'Bearer ' + getToken() // 让每个请求携带自定义token 请根据实际情况自行修改

  }

  // get请求映射params参数

  if (config.method === 'get' && config.params) {

    let url = config.url + '?' + tansParams(config.params);

    url = url.slice(0, -1);

    config.params = {};

    config.url = url;

  }

  if (!isRepeatSubmit && (config.method === 'post' || config.method === 'put')) {

    const requestObj = {

      url: config.url,

      data: typeof config.data === 'object' ? JSON.stringify(config.data) : config.data,

      time: new Date().getTime()

    }

    const requestSize = Object.keys(JSON.stringify(requestObj)).length; // 请求数据大小

    const limitSize = 5 * 1024 * 1024; // 限制存放数据5M

    if (requestSize >= limitSize) {

      console.warn(`[${config.url}]: ` + '请求数据大小超出允许的5M限制,无法进行防重复提交验证。')

      return config;

    }

    const sessionObj = cache.session.getJSON('sessionObj')

    if (sessionObj === undefined || sessionObj === null || sessionObj === '') {

      cache.session.setJSON('sessionObj', requestObj)

    } else {

      const s_url = sessionObj.url;                  // 请求地址

      const s_data = sessionObj.data;                // 请求数据

      const s_time = sessionObj.time;                // 请求时间

      const interval = 1000;                         // 间隔时间(ms),小于此时间视为重复提交

      if (s_data === requestObj.data && requestObj.time - s_time < interval && s_url === requestObj.url) {

        const message = '数据正在处理,请勿重复提交';

        console.warn(`[${s_url}]: ` + message)

        return Promise.reject(new Error(message))

      } else {

        cache.session.setJSON('sessionObj', requestObj)

      }

    }

  }

  return config

}, error => {

    console.log(error)

    Promise.reject(error)

})
  • 响应拦截器在请求得到响应之后,对响应体的一些处理,通常是数据统一处理等,也常用来判断登录失效等。此外,如果服务器返回的状态码超出 2xx 范围,响应拦截器也会进行处理。
service.interceptors.response.use(res => {

    // 未设置状态码则默认成功状态

    const code = res.data.code || 200;

    // 获取错误信息

    const msg = errorCode[code] || res.data.msg || errorCode['default']

    // 二进制数据则直接返回

    if (res.request.responseType ===  'blob' || res.request.responseType ===  'arraybuffer') {

      return res.data

    }

    if (code === 401) {

      if (!isRelogin.show) {

        isRelogin.show = true;

        MessageBox.confirm('登录状态已过期,您可以继续留在该页面,或者重新登录', '系统提示', { confirmButtonText: '重新登录', cancelButtonText: '取消', type: 'warning' }).then(() => {

          isRelogin.show = false;

          store.dispatch('LogOut').then(() => {

            location.href = '/index';

          })

      }).catch(() => {

        isRelogin.show = false;

      });

    }

      return Promise.reject('无效的会话,或者会话已过期,请重新登录。')

    } else if (code === 500) {

      Message({ message: msg, type: 'error' })

      return Promise.reject(new Error(msg))

    } else if (code === 601) {

      Message({ message: msg, type: 'warning' })

      return Promise.reject('error')

    } else if (code !== 200) {

      Notification.error({ title: msg })

      return Promise.reject('error')

    } else {

      return res.data

    }

  },

  error => {

    console.log('err' + error)

    let { message } = error;

    if (message == "Network Error") {

      message = "后端接口连接异常";

    } else if (message.includes("timeout")) {

      message = "系统接口请求超时";

    } else if (message.includes("Request failed with status code")) {

      message = "系统接口" + message.substr(message.length - 3) + "异常";

    }

    Message({ message: message, type: 'error', duration: 5 * 1000 })

    return Promise.reject(error)

  }

)