详解AngularJS中的http拦截


Posted in Javascript onFebruary 09, 2016

http拦截,即$http服务允许我们与服务端交互,有时候我们希望在发出请求之前以及收到响应之后做些事情。
$httpProvider包含了一个interceptors的数组。

我们这样创建一个interceptor。

app.factory('myInterceptor', ['$log', function($log){
  $log.debug('');
  
  var myInterceptor = {};
  
  return myInterceptor;
}])

接着注册interceptor. 

app.config(['$httpProvider', function($httpProvider){
  $httpProvider.interceptors.push('myInterceptor');
}])

以下是$http拦截的一些例子。

■ 拦截器中的异步操作

app.factory('myInterceotpr','someAsyncServcie', function($q, someAsyncServcie){
  var requestInterceptor = {
    request: function(config){
      var deferred = %q.defer();
      someAsyncService.doAsyncOperation().then(function(){
        ...
        deferred.resolve(config);
      }, function(){
        ...
        deferred.resolve(config);
      })
      return deferred.promise;
    }
  };
  
  return requestInterceptor;
})

以上,是一个请求拦截,做了一个异步操作,根据异步操作的结果来更新config。

当然也有响应拦截。

app.factory('myInterceptor',['$q', 'someAsyncService', function($q, someAsyncSercice){
  var responseInterceptor = {
    response: function(response){
      var deferred = $q.defer();
      someAsyncService.doAsyncOperation().then(function(response){
        ...
        deferred.resolve(response);
      }, function(response){
        ...
        deferred.resolve(response);
      })
      return deferred.promise;
    }
  };
  return responseInterceptor;
}])

■ Session拦截,请求拦截

服务端有2种类型的验证,一个是基于cookie的,一种是基于token的。对于基于token验证,当用户登录,获取一个来自服务端的token,这个token在每一次请求时发送给服务端。

创建一个有关session的injector:

app.factory('sessionInjector',['SessionService', function(SessionService){
  var sessionInjector = {
    request: function(config){
      if(!SessionService.isAnonymous){
        config.headers['x-session-token'] = SessionService.token;
      }
      return config;
    }
  };
  
  return sessionInjector;
}])

可见,把从服务端返回的token放在了config.headers中。

注册injector:

app.config(['$httpProvider', function($httpProvider){
  $httpProvider.interceptors.push('sessionInjector');
}])

发出一个请求:

$http.get('');

拦截前大致是:

{
  "transformRequest":[null],
  "transformResponse":[null],
  "method":"GET",
  "url":"",
  "headers":{
    "Accept": "application/json, text/plain,*/*"
  }
}

拦截后,在headers中多两个一个x-session-token字段:

{
  "transformRequest":[null],
  "transformResponse":[null],
  "method":"GET",
  "url":"",
  "headers":{
    "Accept": "application/json, text/plain,*/*",
    "x-session-token":......
  }
}

■ 时间戳,请求和响应拦截

app.factory('timestampMarker',[function(){
  var timestampMarker = {
    request:function(config){
      config.requestTimestamp = new Date().getTime();
      return config;
    },
    response: function(response){
      response.config.responseTimestamp = new Date().getTime();
      return config;
    }
  };
  
  return timestampMarker;
}])

以上,在请求和响应时拦截,在config.requestTimestamp和config.responseTimestamp赋上当前的时间。

注册拦截器:

app.config(['$httpProvider', function($httpProvider){
  $httpProvider.interceptors.push('timestampMarker');
}])

然后在运用的时候可以算出请求响应所耗去的时间。

$http.get('').then(function(response){
  var time = response.config.responseTime - response.config.requestTimestamp;
  console.log('请求耗去的时间为 ' + time);
})

■ 请求错误恢复,请求拦截

模拟一个请求拦截的错误情形:

app.factory('requestRejector',['$q', function($q){
  var requestRejector = {
    request: function(config){
      return $q.reject('requestRejector');
    }
  };
  return requestRejector;
}])

拦截请求错误:

app.factory('requestRecoverer',['$q', function($q){
  var requestRecoverer = {
    requestError: function(rejectReason){
      if(rejectReason === 'requestRejector'){
        //恢复请求
        return {
          transformRequest:[],
          transformResponse:[],
          method:'GET',
          url:'',
          headers:{
            Accept:'application/json, text/plain, */*'
          }
        };
      } else {
        return $q.reject(rejectReason);
      }
    }
  };
  
  return requestRecoverer;
}])

注册拦截器:

app.config(['$httpProvider', function($httpProvider){
  $httpProvider.interceptors.push('requestRejector');
  $httpProvider.interceptors.push('requestRecoverer');
}])

■ Session错误恢复,响应拦截

app.factory('sessionRecoverer',['$q','$injector',function($q, $injector){
 var sessionRecoverer = {
  responseError: function(response){
   //如果Session过期
   if(response.status == 419){
    var SessionService = $injector.get('SessionService');
    var $http = $injector.get('$http');
    var deferred = $q.defer();
    
    //创建一个新的session
    SessionService.login().then(deferred.resolve, deferred.reject);
    
    return deferred.promise.then(function(){
     reutrn $http(response.config);
    })
   }
   return $q.reject(response);
  }
 };
 
 return sessionRecoverer;
}])

以上就是本文的全部内容,希望对大家的学习有所帮助。

Javascript 相关文章推荐
JQuery团队打造的javascript单元测试工具QUnit介绍
Feb 26 Javascript
jquery取消选择select下拉框示例代码
Feb 22 Javascript
浅谈angularJS中的事件
Jul 12 Javascript
js 动态生成json对象、时时更新json对象的方法
Dec 02 Javascript
Vue 单文件中的数据传递示例
Mar 21 Javascript
js实现音频控制进度条功能
Apr 01 Javascript
详解vue-cli + webpack 多页面实例配置优化方法
Jul 13 Javascript
如何在vue里添加好看的lottie动画
Aug 02 Javascript
react基本安装与测试示例
Apr 27 Javascript
谈一谈vue请求数据放在created好还是mounted里好
Jul 27 Javascript
关于vue的列表图片选中打钩操作
Sep 09 Javascript
vue实现div可拖动位置也可改变盒子大小的原理
Sep 16 Javascript
详解Javacript和AngularJS中的Promises
Feb 09 #Javascript
深入浅析JavaScript面向对象和原型函数
Feb 06 #Javascript
基于JavaScript实现图片点击弹出窗口而不是保存
Feb 06 #Javascript
javascript+css3 实现动态按钮菜单特效
Feb 06 #Javascript
Angularjs全局变量被作用域监听的正确姿势
Feb 06 #Javascript
JavaScript仿商城实现图片广告轮播实例代码
Feb 06 #Javascript
简介AngularJS中$http服务的用法
Feb 06 #Javascript
You might like
php中实现可以返回多个值的函数实例
2015/03/21 PHP
PHP+shell实现多线程的方法
2015/07/01 PHP
php实现scws中文分词搜索的方法
2015/12/25 PHP
PHP中strpos、strstr和stripos、stristr函数分析
2016/06/11 PHP
php blowfish加密解密算法
2016/07/02 PHP
从ThinkPHP3.2.3过渡到ThinkPHP5.0学习笔记图文详解
2019/04/03 PHP
深入理解JavaScript 闭包究竟是什么
2013/04/12 Javascript
JS对象转换为Jquery对象示例
2014/01/26 Javascript
现代 JavaScript 开发编程风格Idiomatic.js指南中文版
2014/05/28 Javascript
基于javascript实现tab选项卡切换特效调试笔记
2016/03/30 Javascript
判断JS对象是否拥有某属性的方法推荐
2016/05/12 Javascript
Bootstrap CSS组件之面包屑导航(breadcrumb)
2016/12/17 Javascript
详解NODEJS的http实现
2018/01/04 NodeJs
nginx部署访问vue-cli搭建的项目的方法
2018/02/12 Javascript
vue页面切换过渡transition效果
2018/10/08 Javascript
JavaScript键盘事件常见用法实例分析
2019/01/03 Javascript
详解使用webpack+electron+reactJs开发windows桌面应用
2019/02/01 Javascript
基于mpvue搭建微信小程序项目框架的教程详解
2019/04/10 Javascript
jQuery 函数实例分析【函数声明、函数表达式、匿名函数等】
2020/05/19 jQuery
在vant 中使用cell组件 定义图标该图片和位置操作
2020/11/02 Javascript
Python利用公共键如何对字典列表进行排序详解
2018/05/19 Python
对python实现二维函数高次拟合的示例详解
2018/12/29 Python
对python中矩阵相加函数sum()的使用详解
2019/01/28 Python
在Python 字典中一键对应多个值的实例
2019/02/03 Python
详解python--模拟轮盘抽奖游戏
2019/04/12 Python
Python 2种方法求某个范围内的所有素数(质数)
2020/01/31 Python
阿玛尼化妆品美国官网:Giorgio Armani Beauty
2017/02/02 全球购物
学生安全责任书
2014/04/15 职场文书
导师就业推荐信范文
2014/05/22 职场文书
售后客服工作职责
2014/06/16 职场文书
学习十八届四中全会精神思想汇报
2014/10/23 职场文书
文明班级申报材料
2014/12/24 职场文书
2015纪念九一八事变84周年演讲稿
2015/03/19 职场文书
2015年党建工作目标责任书
2015/05/08 职场文书
博物馆观后感
2015/06/05 职场文书
win10搭建配置ftp服务器的方法
2022/08/05 Servers