zoukankan      html  css  js  c++  java
  • AngularJS Resource:与 RESTful API 交互

    REST(表征性状态传输,Representational State Transfer)是Roy Fielding博士在2000年他的博士论文中提出来的一种软件架构风格。RESTful风格的设计不仅具有更好的可读性(Human Readable),而且易于做缓存以及服务器扩展(scalability)。REST风格体现在URL设计上:

    • 每个URL对应一个资源
    • 对资源的不同操作对应于HTTP的不同方法
    • 资源表现形式(representation)通过AcceptContent-Type指定

    AngularJS提供了$resourceService来更方便地与RESTful服务器API进行交互,可以方便地定义一个REST资源,而不必手动所有的声明CRUD方法。

    参考文档: https://docs.angularjs.org/api/ngResource/service/$resource

    Resource Factory

    $resourceService定义在ngResourceModule中,需要在你的HTML中引入这个Module对应的JS,同时在你的APP中添加这样一个依赖:

    var app = angular.module('helloApp, ['ngResource']);
    

    然后为资源建立一个Factory:

    app.factory('Notes', ['$resource', function($resource) {
        return $resource('/notes/:id');
    }]);
    

    当然,你也可以不把$esource的实例放到Factory里,直接在控制器中存起来:var Notes = $resource('/notes/:id)

    CRUD

    在你的控制器中就可以对资源进行增删改查了:

    app.controller('NotesCtrl', ['$scope', 'Notes', function($scope, Notes) {
        var notes = Notes.query(function(){
            // GET: /notes
            // Response: [{id: 1, content: 'hello'}, {id: 2, content: 'world'}];
    
            var first = notes[0];
            first.content = 'halo';
            first.$save();
            // POST: /notes/1 {id: 1, content: 'halo'}
            // Response: {id: 1, content: 'halo'}
    
            second.$delete();
            // DELETE: /notes/2
        });
    
        var note = new Notes({content: 'xxx'});
        note.$save();
        // POST: /notes
        // Response: {id: 3, content: 'xxx'}
    }]);
    

    PUT 操作

    $resource提供了五种默认操作:getquerysaveremovedelete。你可以配置一个update操作来完成HTTP PUT:

    app.factory('Notes', ['$resource', function($resource) {
        return $resource('/notes/:id', null, {
            update: { method:'PUT' }
        }); 
    }]);
    

    现在,你可以在控制器中获取一个note并更新它:

    var note = Notes.get({ id: 3}),
        $id = note.id;
    
    note.content = 'yyy';
    Notes.update({ id:$id }, note);
    // PUT /notes/3 {id: 3, content: 'yyy'}
    

    现在你的Notes有六种操作了。这些操作有两种调用方式:

    1. 通过资源类调用,例如:Notes.update({id: xxx})
    2. 通过资源实例调用,例如:note.$update(),此时操作名需加前缀$

    具体的调用参数可参考文档:

    HTTP GET "class" actions: Resource.action([parameters], [success], [error])

    non-GET "class" actions: Resource.action([parameters], postData, [success], [error])

    non-GET instance actions: instance.$action([parameters], [success], [error])

    其中,success参数为(value, responseHeaders),error参数为(httpResponse)

    属性/URL映射

    上述例子中,我们看到note对象的id属性会映射到URL中的:id/notes/:id)。如果你的业务更加复杂,可以手动配置这个映射关系。例如:

    var Notes = $resouce('/users/:userId/notes/:noteId', {
        noteId: '@id',
        userId: '@owner'
    }
    

    将会读取noteownerid属性来生成URL,比如删除note时:

    // note === {id: 123, owner: 'alice', content: 'hello'}
    note.$delete();
    // DELETE: /users/alice/notes/123
    

    在构造$resource时,多于的属性映射会成为URL Query。例如:

    var Notes = $resouce('/notes/:id', {
        id: '@id',
        user: '@owner'
    });
    // note === {id: 123, owner: 'alice', content: 'hello'}
    note.$delete();
    // DELETE: /notes/123?user=alice
    

    REST操作的声明和调用中,多于的属性会成为URL Query。例如:

    var Notes = $resouce('/notes/:id', {id: '@id'}, {
        update: {method: 'PUT', operator: 'bob'}
    });
    // note === {id: 123, content: 'hello'}
    note.$update({trusted: true});
    // PUT: /notes/123?operator=bob&trusted=true {id: 123, content: 'hello'}
    

    响应转换

    有时基于既定的后台设计,无法提供完全RESTful的API,比如/notes返回的是一个分页器对象,而非数组。此时,我们仍然可以使用$resource,但需要设置响应转换回调。例如:

    var Notes = $resouce('/notes/:id', null, {
        pager: {
            method: 'GET', 
            transformResponse: function(data, headers){
                // Server respond: 
                // data = {currentPage: 1, 
                //     totalPage: 20, 
                //     pageSize: 2,
                //     content: [{id: 1, content: 'hello'}, {id: 2, content: 'world'}]}
                var pager = JSON.parse(data);
                return pager.content;
            }
        }
    });
    var notes = Notes.query(function(){
        // GET: /notes
        // notes === [{id: 1, content: 'hello'}, {id: 2, content: 'world'}]
    });
    

    类似响应重写,你还可以设置请求转换transformRequest

    虽然$resource的设计可以支持绝大多数的URL和内容表示设计,但如果你发现$resource的使用过程极其复杂,那可能是你的服务器API并不满足RESTful风格。

    我喜欢程序员,他们单纯、固执、容易体会到成就感;面对压力,能够挑灯夜战不眠不休;面对困难,能够迎难而上挑战自我。他 们也会感到困惑与傍徨,但每个程序员的心中都有一个比尔盖茨或是乔布斯的梦想“用智慧开创属于自己的事业”。我想说的是,其 实我是一个程序员
  • 相关阅读:
    测试用例设计方法
    软件测试--常考的证书
    fillder-----接口延时返回(新方法)
    测试工作之--adb代码
    设计测试用例的常用方法
    TFS2013 设置签出独占锁
    Sql Server (MSSQLSERVER) 服务无法启动
    Sql 四大排名函数(ROW_NUMBER、RANK、DENSE_RANK、NTILE)简介
    C#使用redis学习笔记
    C#使用mybatis学习笔记
  • 原文地址:https://www.cnblogs.com/kms1989/p/5653306.html
Copyright © 2011-2022 走看看