设计指南及请求返回规范

RESTful API 设计指南及请求返回规范

当前发展趋势,前后端逐渐分离成不同的项目,前端负责页面路由和页面渲染,后端通过API接口提供数据支持。因此,必须有一种统一的机制,方便不同的前端设备与后端进行通信。这导致API构架的流行,甚至出现"API First"的设计思想。RESTful API是目前比较成熟的一套互联网应用程序的API设计理论。

一、什么是RESTful?

1.1 域名

一般情况下,放置跨域调用,API接口放在项目本域名下:

https://example.com/api/

1.2 路径(Endpoint)

路径又称"终点"(endpoint),表示API的具体网址。

在RESTful架构中,每个网址代表一种资源(resource),所以网址中不能有动词,只能有名词,而且所用的名词往往与数据库的表格名对应。一般来说,数据库中的表都是同种记录的"集合"(collection),所以API中的名词也应该使用复数

举例来说,有一个API提供动物园(zoo)的信息,还包括各种动物和雇员的信息,则它的路径应该设计成下面这样。

https://example.com/api/zoos
https://example.com/api/animals
https://example.com/api/employees

1.3 HTTP动词

对于资源的具体操作类型,由HTTP动词表示。

常用的HTTP动词有下面五个(括号里是对应的SQL命令)。

GET(SELECT):从服务器取出资源(一项或多项)。
POST(CREATE):在服务器新建一个资源。
PUT(UPDATE):在服务器更新资源(客户端提供改变后的完整资源)。
PATCH(UPDATE):在服务器更新资源(客户端提供改变的属性)。
DELETE(DELETE):从服务器删除资源。

还有两个不常用的HTTP动词。

HEAD:获取资源的元数据。
OPTIONS:获取信息,关于资源的哪些属性是客户端可以改变的。

下面是一些例子。

GET /zoos:列出所有动物园
POST /zoos:新建一个动物园
GET /zoos/ID:获取某个指定动物园的信息
PUT /zoos/ID:更新某个指定动物园的信息(提供该动物园的全部信息)
PATCH /zoos/ID:更新某个指定动物园的信息(提供该动物园的部分信息)
DELETE /zoos/ID:删除某个动物园
GET /zoos/ID/animals:列出某个指定动物园的所有动物
DELETE /zoos/ID/animals/ID:删除某个指定动物园的指定动物

1.4 过滤信息(Filtering)

如果记录数量很多,服务器不可能都将它们返回给用户。API应该提供参数,过滤返回结果。

下面是一些常见的参数。

?size=10:指定返回记录的数量
?page=0:指定返回记录的开始位置。
?page=2&size=100:指定第几页,以及每页的记录数。
?sort=name,asc:指定返回结果按照哪个属性排序,以及排序顺序。
?animal_type_id=1:指定筛选条件

参数的设计允许存在冗余,即允许API路径和URL参数偶尔有重复。比如,GET /zoo/ID/animals 与 GET /animals?zoo_id=ID 的含义是相同的。

1.5 状态码(Status Codes)

服务器向用户返回的状态码和提示信息,常见的有以下一些(方括号中是该状态码对应的HTTP动词)。

200 OK - [GET]:服务器成功返回用户请求的数据,该操作是幂等的(Idempotent)。
201 CREATED - [POST/PUT/PATCH]:用户新建或修改数据成功。
202 Accepted - [*]:表示一个请求已经进入后台排队(异步任务)
204 NO CONTENT - [DELETE]:用户删除数据成功。
400 INVALID REQUEST - [POST/PUT/PATCH]:用户发出的请求有错误,服务器没有进行新建或修改数据的操作,该操作是幂等的。
401 Unauthorized - [*]:表示用户没有权限(令牌、用户名、密码错误)。
403 Forbidden - [*] 表示用户得到授权(与401错误相对),但是访问是被禁止的。
404 NOT FOUND - [*]:用户发出的请求针对的是不存在的记录,服务器没有进行操作,该操作是幂等的。
406 Not Acceptable - [GET]:用户请求的格式不可得(比如用户请求JSON格式,但是只有XML格式)。
410 Gone -[GET]:用户请求的资源被永久删除,且不会再得到的。
422 Unprocesable entity - [POST/PUT/PATCH] 当创建一个对象时,发生一个验证错误。
500 INTERNAL SERVER ERROR - [*]:服务器发生错误,用户将无法判断发出的请求是否成功。

二、RESTful风格示例

2.1 RESTful风格接口设计

在Spring MVC 中,RESTful API的定义对应为Controller层。这里使用一个对用户(Users)常见接口来说明Restful风格的设计规范。前面介绍过的RESTful的接口定义规范:

GET(SELECT):从服务器取出资源(一项或多项)。
POST(CREATE):在服务器新建一个资源。
PUT(UPDATE):在服务器更新资源(客户端提供改变后的完整资源)。
PATCH(UPDATE):在服务器更新资源(客户端提供改变的属性)。
DELETE(DELETE):从服务器删除资源。

我们按照规范设计如下的接口:

@RestController
public class UserController {


    @Autowired
    private UserRepository userRepository;

    /**
     * 获取用户列表
     */
    @GetMapping("/users")
    @ResponseStatus(HttpStatus.OK)
    public List<Users>  getUsers(){
        return userRepository.findUserList();
    }

    /**
     * 创建一个用户
     */
    @PostMapping(value = "/users")
    @ResponseStatus(HttpStatus.CREATED)
    public Object addUser(@RequestBody Users users){
        return userRepository.save(users);
    }

    /**
     * 根据用户id获取用户信息
     */
    @GetMapping(value = "/users/{id}")
    @ResponseStatus(HttpStatus.OK)
    public Object getUser(@PathVariable("id") String id) throws NotFoundException
    {
        return userRepository.getUserById(id);
    }

    /**
     * 根据用户id删除一个用户
     */
    @DeleteMapping(value = "/users/{id}")
    @ResponseStatus(HttpStatus.NO_CONTENT)
    public void deleteUser(@PathVariable("id") String id)
    {
        userRepository.deleteUser(id);
    }

    /**
     * 根据用户id修改用户信息
     */
    @PutMapping(value = "/users/{id}")
    @ResponseStatus(HttpStatus.CREATED)
    public Users updateUser(@PathVariable("id") String id, @RequestBody Users user)
    {
        return userRepository.update(id, user);
    }

}

在这些接口上,使用了如下注解:

  • @GetMapping
  • @PostMapping
  • @DeleteMapping
  • @PutMapping

Spring 4.3之后推荐使用这种写法,旨在简化常用的HTTP方法的映射,并可以更好地表达被注解方法的语义。如@GetMapping实际上是一个组合注解,可以直接代替@RequestMapping(method = RequestMethod.GET),其他同理。

同时,我们在每个接口上显示得使用了注解@ResponseStatus,用来标识接口正常返回时的HTTP状态码。另外,我们还需要注意每个接口的返回结果,除了删除用户,其他每个接口都有返回值。这是因为RESTful 规范中提到:

GET /collection:返回资源对象的列表(数组)
GET /collection/resource:返回单个资源对象
POST /collection:返回新生成的资源对象
PUT /collection/resource:返回完整的资源对象
PATCH /collection/resource:返回完整的资源对象
DELETE /collection/resource:返回一个空文档

2.2 关于PUT和PATCH

根据RESTful的思想,我们知道更新操作可以分为全部更新和部分更新。结合HTTP语义,可以表示为:

PUT /users/{id}:更新某个人的信息(提供此人的全部信息)
PATCH /users/{id}:更新某个人的信息(提供此人的部分信息,比如只修该age字段)

但实际上,PATCH语义的应用并不广泛。所以,为了方便,我将两个接口合在一起,同时支持全部和部分更新,HTTP动词使用PUT,仅供参考,大家酌情而定。

三、返回信息结构

前后端分离之后,前后端开发人员依靠接口文档进行开发,这时候规范的返回参数结构就变得尤为重要,约定好返回参数将大大提升开发效率,下面对返回参数进行规范说明。

以下涉及所有参考代码,可点我查看

3.1 所有正确的返回JSON

状态码为2xx

  • 集合对象返回的结构规范
{
    "total": 200,  // 符合条件数据集合的总条目数
    "data": [{
                   // 符合条件当前分页的条件列表
    }]     
}

这里需要定义一个用于返回的值对象:

/**
 * 请求返回集合的封装值对象
 */
public class ResponseCollection {

    private Long total;

    private Object data;

    
    setter and getter omit...
}

改造controller层相应方法:


    /**
     * 获取用户列表
     */
    @GetMapping("/users")
    @ResponseStatus(HttpStatus.OK)
    public ResponseCollection getUsers(Pageable pageable){
        Page<Users> result= userRepository.findUserList(pageable);
        //将结果封装到集合类结果的值对象中
        ResponseCollection resp= new ResponseCollection();
        resp.setTotal(result.getTotalElements());
        resp.setData(result.getContent());
        return resp;
    }

将的数据组装后返回,结合上面的示例,对获取用户列表接口进行请求:

localhost:8080/users?page=0&size=10

返回集合对象,结构如下:

{
    "total": 2,
    "data": [
        {
            "id": "bc63312b-0536-412c-9d36-54614002d32a",
            "userName": "张三",
            "age": 25
        },
        {
            "id": "ec7201d8-5c9f-4fa7-9a83-4a1f92192bc1",
            "userName": "李四",
            "age": 18
        }
    ]
}
  • 单个对象返回的结构规范
{
    //具体对象内容
}

单个对象的查询结果无需封装直接返回即可,对根据id获取用户详情接口进行请求:

localhost:8080/users/bc63312b-0536-412c-9d36-54614002d32a

返回单个对象,结构够如下:

{
    "id": "bc63312b-0536-412c-9d36-54614002d32a",
    "userName": "张三",
    "age": 25
}

3.2 有错误时候的返回JSON

服务器异常的结构规范:

{
    "status": 500,
    "error": "INTERNAL SERVER ERROR",
    "message": "服务器发生错误"
}

参数错误的结构规范:

{
    "status": 422,
    "error": "UNPROCESABLE ENTITY",
    "message": "提交的数据格式有错误",
    "details": [{
        "field": "email",                  // 有格式验证错误的属性名
        "message": "请输入正确的email地址"  // 详细的错误信息
    }]
}

对于错误时候的返回,应当兼容http的状态码,尽量避免出现返回状态是200,而在返回内容里表明存在错误。其实spring boot提供了一种更好的解决方方案,可以使用Spring 全局异常处理机制,我们可以通过使用@ControllerAdvice注解定义全局统一的异常处理类来完成需求。也即是说,在处理错误时,我们不再直接返回一个封装的值对象,而采用异常机制。下面介绍一下实现方式。

首先定义一个处理参数异常类,该类专门处理非法参数的异常:


/**
 * 自定义参数异常类
 */
public class ParamsException extends Exception{

    private List<Map<String,String>> details;


    public ParamsException(String message, List<Map<String, String>> details) {
        super(message);
        this.details = details;
    }

    setter and getter omit...
}

新建类RestExceptionHandler用于处理全局异常,使用注解@ControllerAdvice,如下:

/**
 * 全局异常处理
 */
@ControllerAdvice
public class RestExceptionHandler {

    /**
     * 处理参数异常
     */
    @ExceptionHandler(value = ParamsException.class)
    @ResponseBody
    @ResponseStatus(HttpStatus.UNPROCESSABLE_ENTITY)
    public ResponseParamsError handleParamsException(ParamsException e)
    {
        return new ResponseParamsError(422,e.getMessage(),"UNPROCESABLE ENTITY",e.getDetails());
    }
}

如上,通过使用注解@ControllerAdvice,类RestExceptionHandler就可以实现全局异常的拦截处理功能。自定义的方法handleParamsException旨在拦截ParamsException异常,一旦拦截成功后,我们可以进行各种处理操作,并且返回自己想要的结果。

其中,注解@ExceptionHandler表示要拦截的异常;注解@ResponseStatus可以指定HTTP响应的状态码;当然,注解@ResponseBody也必不可少。

这里定义了一个返回参数异常信息的值对象,用于全局异常处理进行参数处理后对结果进行封装:

/**
 * 用于返回参数异常的值对象
 */
public class ResponseParamsError {

    private int status;

    private String message;

    private String error;

    private List<Map<String,String>> details;


    public ResponseParamsError(int status, String message, String error, List<Map<String, String>> details) {
        this.status = status;
        this.message = message;
        this.error = error;
        this.details = details;
    }

    setter and getter omit...
}

  • 下面对这个错误类型进行测试:

新增一个接口用于测试参数错误异常:


    /**
     * 根据用户年龄获取大于该年龄的用户集合
     */
    @GetMapping(value = "/users/age/{age}")
    @ResponseStatus(HttpStatus.OK)
    public ResponseCollection getUsersOlderThan(@PathVariable("age") Integer age,Pageable pageable) throws ParamsException {
        if (age>120){
            List<Map<String,String>> details= new ArrayList<>();
            Map<String,String> map = new HashMap<>();
            map.put("failed","age");
            map.put("message","年龄不能超过120岁");
            details.add(map);
            throw new ParamsException("提交的数据格式有误",details);
        }else {
            Page<Users> result= userRepository.getUsersOlderThan(age,pageable);
            //将结果封装到集合类结果的值对象中
            ResponseCollection resp= new ResponseCollection();
            resp.setTotal(result.getTotalElements());
            resp.setData(result.getContent());
            return resp;
        }
    }

可以看出,该接口是根据用户年龄获取大于该年龄的用户集合。对请求参数进行校验是非常常见的业务逻辑,就本接口而言,当年龄的参数大于120的时候,我们抛出一个参数异常(ParamsException),交个全局处理异常区处理。

向该接口发送请求数据:

localhost:8080/users/age/230

看该请求,年龄参数age大于120,此参数有误,返回错误信息,结构如下:

{
    "status": 422,
    "message": "提交的数据格式有误",
    "error": "UNPROCESABLE ENTITY",
    "details": [
        {
            "failed": "age",
            "message": "年龄不能超过120岁"
        }
    ]
}

3.3 需要前端路由跳转的

{
    "status": 302,
    "data": "http://open.wx.qq.com/oauth2/connect?appid=xdfdsddffs&ddd=xxxx"
}

注:转载请注明出处

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

推荐阅读更多精彩内容