Angular学习第九天

$http 服务是什么?

$http 就是JQuery中的$.ajax

如何用?

请求一个已存在的json文件

news.json

{
    "status": "1",
    "message": "成功",
    "data": [
        {
            "id": "556",
            "title": "left_top秘密任务",
            "image": "/resources/news/1479110226.png",
            "content": "",
            "url": "/challenge",
            "desc": "",
            "created_at": "2016-06-27 11:25:29"
        }, {
            "id": "555",
            "title": "right_top晒勋章",
            "image": "/resources/news/1476350699.png",
            "content": "",
            "url": "/medal",
            "desc": "",
            "created_at": "2016-06-27 11:23:56"
        }, {
            "id": "554",
            "title": "left_bottom邀请有礼",
            "image": "/resources/news/1476181116.png",
            "content": "",
            "url": "/invest_timeline",
            "desc": "",
            "created_at": "2016-05-17 17:40:05"
        }, {
            "id": "496",
            "title": "right_bottom新手教程",
            "image": "/resources/news/1476181156.png",
            "content": "",
            "url": "http://mp.weixin.qq.com/s?__biz=MzA5ODI4MjM1NA==&mid=2649936983&idx=2&sn=ebdaaa65f2216ff223973d25329339f5",
            "desc": "",
            "created_at": "2016-03-17 10:33:01"
        }
    ]
}

controller

//初始我们定义一个 $scope.news 请求成功之后我们给这个变量重新赋值
appH5.controller("myTabCtrl",['$scope','$http',function($scope,$http){
  $scope.news=[];
  $http.get("mock/news.json")
  .success(function(res){
    console.log(res);
    if(res.status==1){
      $scope.news=res.data;
    }else{
      alert(res.message);
    }
  }).error(function(){
    console.log("error");
  }).then(function(data){
    console.log(data);
  });
});
$http服务会返回一个 promise;你可以定义success方法说明请求成功之后执行什么;同样你也可以定义error方法说明请求成功之后执行什么。
请求get请求的接口
// 请求成功之后我们给 orders 这个变量重新赋值
appH5.controller("myTabCtrl",['$scope','$http',function($scope,$http){
  $http.get("http://www.mxx.com/test.php?c=orders")
  .success(function(res){
    console.log(res);
    if(res.status==1){
      $scope.orders=res.data;
    }else{
      alert(res.message);
    }
  }).error(function(){
    console.log("error");
  });
});
请求post请求的接口
// 请求成功之后我们给 orders 这个变量赋值
appH5.controller("myTabCtrl",['$scope','$http',function($scope,$http){
  $http({
    method:'post',
    url:'http://www.mxx.com/test.php?c=login',
    data:$.param({name:"aaa",id:1,age:20}),
    headers:{'Content-Type': 'application/x-www-form-urlencoded'}
  }).success(function(req){
    console.log(res);
    if(res.status==1){
      $scope.orders=res.data;
    }else{
      alert(res.message);
    }
  })
});
  • $.param方法是jQuery的序列化方法,post的用法需要借助这个类库;并且需要设置headers:{'Content-Type': 'application/x-www-form-urlencoded'}

用$http发送get请求

appH5.controller("myTabCtrl",['$scope','$http',function($scope,$http){
  $http({
    method: 'GET',
    url: 'http://www.mxx.com/test.php',
    params: { c: 'orders' },//params作为url的参数
  }).success(function(req){
  console.log(req);
  });
});

//最后请求的url实际上是 
//http://www.mxx.com/test.php?c=orders;
//params定义的参数会被拼接在url后面

$http发送POST请求

appH5.controller("myTabCtrl",['$scope','$http',function($scope,$http){
  $http({
    method:'post',
    url:'http://www.mxx.com/test.php?c=login',
    data:$.param({name:"aaa",id:1,age:20}),
    headers:{'Content-Type': 'application/x-www-form-urlencoded'}
  }).success(function(req){
  console.log(req);
  });
});

如果你需要在请求完成得到返回数据之后执行某段操作的话,你可以用 then()方法让代码顺序执行

appH5.controller("myTabCtrl",['$scope','$http',function($scope,$http){
  $http({
    method:'post',
    url:'http://www.mxx.com/test.php?c=login',
    data:$.param({name:"aaa",id:1,age:20}),
    headers:{'Content-Type': 'application/x-www-form-urlencoded'}
  }).success(function(req){
  console.log(req);
  }).then(function(res){
    console.log("我是得到返回数据之后才会执行的操作");
  });
});
  • 为什么post请求我要用 $.param?

我们来看一下我们不用$.param把data序列化,我们在调试工具中看到的传递到后台的参数

2180072-a84b93de0f80bdbc.png

当我们使用 jQuery的 ajax 方法,我们看到的参数是

2180072-901cbeca165d8aed.png

jQuery的 ajax 方法

  • 解释 看起来执行请求的代码是一样的,但是实际发送的时候明显后者是键值对的方式,前者则不是;为什么会出现这种问题呢?因为jQuery在发送 data里面的数据之前会进行数据的序列化(Content-Type:application/x-www-form-urlencoded)如
var data= {name:"aaa",id:1,age:20};
// jQuery在post数据之前会把data转换成字符串:"name=aaa&id=1&age=20"

相反的 Angular不会执行这一操作,它,默认传递的json字符串(Content-Type: application/json;),后端在取前端传递的参数时不能用 $_REQUEST/$_POST(本人只会PHP,这里以php为例),而必须用

$params = json_decode(file_get_contents('php://input'),true);

这种方式来获取前端的参数。这就意味着你一旦你之前用jQuery写的项目要用Angular重构,那么后端的接口就必须一个一个更改获取参数的方式;

  • 如果不用后端改,有没有解决方案?
  • 方案1,其实我已经写出来了,就是用$.param函数来序列化之后再赋值给$http 的data属性,不过你得为了这个引入jQuery这个类库。(当然如果你想写的话,你自己可以写一个序列化函数)

$http({
  method:'post',
  url:'http://www.mxx.com/test.php?c=login',
  data:$.param({name:"aaa",id:1,age:20}),
  headers:{'Content-Type': 'application/x-www-form-urlencoded'}
}).success(function(req){
console.log(req);
})
  • 方案2 修改Angular的$httpProvider的默认配置,这是最初我进我们公司之后用的方法
angular.module('MyModule', [], function($httpProvider) {
// 设置 Content-Type
$httpProvider.defaults.headers.post['Content-Type'] = 'application/x-www-form-urlencoded;charset=utf-8';

/**
 * 序列化data
 * @param {Object} obj
 * @return {String}
 */ 
var param = function(obj) {
  var query = '', name, value, fullSubName, subName, subValue, innerObj, i;

  for(name in obj) {
    value = obj[name];

    if(value instanceof Array) {
      for(i=0; i<value.length; ++i) {
        subValue = value[i];
        fullSubName = name + '[' + i + ']';
        innerObj = {};
        innerObj[fullSubName] = subValue;
        query += param(innerObj) + '&';
      }
    }
    else if(value instanceof Object) {
      for(subName in value) {
        subValue = value[subName];
        fullSubName = name + '[' + subName + ']';
        innerObj = {};
        innerObj[fullSubName] = subValue;
        query += param(innerObj) + '&';
      }
    }
    else if(value !== undefined && value !== null)
      query += encodeURIComponent(name) + '=' + encodeURIComponent(value) + '&';
  }

  return query.length ? query.substr(0, query.length - 1) : query;
};

// 重写 $http 服务的 transformRequest 方法对用户传递过来的数据进行序列化处理
$httpProvider.defaults.transformRequest = [function(data) {
  return angular.isObject(data) && String(data) !== '[object File]' ? param(data) : data;
}];
});

注 这只是简单用法,细节我会稍后编辑,比如为什么post必须用$.param;和设置 header

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,383评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,522评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,852评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,621评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,741评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,929评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,076评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,803评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,265评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,582评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,716评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,395评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,039评论 3 316
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,798评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,027评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,488评论 2 361
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,612评论 2 350

推荐阅读更多精彩内容