来源:点击进入

点击上方链接,版面更好

一、什么是RESTful

REST 是面向资源的,这个概念非常重要,而资源是通过 URI 进行暴露,URI 的设计只要负责把资源通过合理方式暴露出来就可以了,对资源的操作与它无关,操作是通过 HTTP动词来体现。所以REST 通过 URI 暴露资源时,会强调不要在 URI 中出现动词,而是对一类资源只提供一个url,通过GET、POST、PUT、DELETE请求来指定要执行的操作。

非RESTful 用法(多个url且url中存在动词)

http://127.0.0.1/order/query/1 GET  根据订单id查询订单数据
    http://127.0.0.1/order/save POST 新增订单
    http://127.0.0.1/order/update POST 修改订单信息
    http://127.0.0.1/order/delete GET/POST 删除订单信息

RESTful 用法(同一个url,具体的操作通过HTTP请求方式来指定)

http://127.0.0.1/order/1 GET  根据订单id查询订单数据
    http://127.0.0.1/order  POST 新增订单
    http://127.0.0.1/order  PUT 修改订单信息
    http://127.0.0.1/order  DELETE 删除订单信息

REST很好地利用了HTTP本身就有的一些特征,如HTTP动词、HTTP状态码、HTTP报头等等。
        REST API 是基于 HTTP的,所以你的API应该去使用 HTTP的一些标准。这样所有的HTTP客户端(如浏览器)才能够直接理解你的API。

REST返回值是标准的,我们不用单独定义和封装返回的状态码,而是直接使用HTTP的状态码,非RESTful 返回举例:

{
      "code": "0",
      "msg": "成功"
    }

{
      "code": "1",
      "msg": "失败"
    }

这种方式还要我们自己去解析,还要前端和后端去协商你返回的0是啥意思。
二、RESTful 的关键

RESTful的关键是定义可表示流程元素/资源的对象。在REST中,每一个对象都是通过URL来表示的,对象用户负责将状态信息打包进每一条消息内,以便对象的处理总是无状态的。前后端分离的项目基本上就是无状态的,我们经常通过签名判断当前的请求是否合法。

所谓无状态的,即所有的资源,都可以通过URI定位,而且这个定位与其他资源无关,也不会因为其他资源的变化而改变。

举个简单的例子说明一下有状态和无状态的区别,如查询员工的工资,如果查询工资是需要登录系统,进入查询工资的页面,执行相关操作后,获取工资的多少,则这种情况是有状态的,因为查询工资的每一步操作都依赖于前一步操作,只要前置操作不成功,后续操作就无法执行;如果输入一个url即可得到指定员工的工资,则这种情况是无状态的,因为获取工资不依赖于其他资源或状态,且这种情况下,员工工资是一个资源,由一个url与之对应,可以通过HTTP中的GET方法得到资源,这是典型的RESTful风格。

三、RESTful API代码示例

package com.ldy.sboot.demo.controller;
     
    import org.springframework.beans.factory.annotation.Autowired;
    import org.springframework.http.HttpStatus;
    import org.springframework.http.ResponseEntity;
    import org.springframework.web.bind.annotation.DeleteMapping;
    import org.springframework.web.bind.annotation.GetMapping;
    import org.springframework.web.bind.annotation.PathVariable;
    import org.springframework.web.bind.annotation.PostMapping;
    import org.springframework.web.bind.annotation.PutMapping;
    import org.springframework.web.bind.annotation.RequestMapping;
    import org.springframework.web.bind.annotation.RequestParam;
    import org.springframework.web.bind.annotation.RestController;
     
    import com.ldy.sboot.demo.entity.OrderEntity;
    import com.ldy.sboot.demo.service.OrderService;
     
    @RequestMapping("restful/order")
    @RestController
    public class OrderController {
     
        @Autowired
        private OrderService orderService;
     
        /**
         * @描述: 根据ID查询<br>
         * @param id
         * @return
         */
        @GetMapping(value = "{id}")
        //@RequestMapping(method = RequestMethod.GET)
        public ResponseEntity<OrderEntity> queryOrderById(@PathVariable("id") Long id) {
            try {
                OrderEntity entity = orderService.queryOrderById(id);
                if (null == entity) {
                    // 资源不存在,响应404
                    return ResponseEntity.status(HttpStatus.NOT_FOUND).body(null);
                }
                // 200
                // return ResponseEntity.status(HttpStatus.OK).body(entity);
                return ResponseEntity.ok(entity);
            } catch (Exception e) {
                e.printStackTrace();
            }
            // 500
            return ResponseEntity.status(HttpStatus.INTERNAL_SERVER_ERROR).body(null);
        }
     
        /**
         * @描述: 新增<br>
         * @param entity
         * @return
         */
        @PostMapping
        //@RequestMapping(method = RequestMethod.POST)
        public ResponseEntity<Void> saveOrder(OrderEntity entity) {
            try {
                orderService.saveOrder(entity);
                return ResponseEntity.status(HttpStatus.CREATED).build();
            } catch (Exception e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }
            // 500
            return ResponseEntity.status(HttpStatus.INTERNAL_SERVER_ERROR).body(null);
        }
     
        /**
         * @描述: 修改<br>
         * @param entity
         * @return
         */
        @PutMapping
        //@RequestMapping(method = RequestMethod.PUT)
        public ResponseEntity<Void> updateOrder(OrderEntity entity) {
            try {
                orderService.updateOrder(entity);
                return ResponseEntity.status(HttpStatus.NO_CONTENT).build();
            } catch (Exception e) {
                e.printStackTrace();
            }
            // 500
            return ResponseEntity.status(HttpStatus.INTERNAL_SERVER_ERROR).body(null);
        }
     
        /**
         * @描述: 删除<br>
         * @param id
         * @return
         */
        @DeleteMapping
        //@RequestMapping(method = RequestMethod.DELETE)
        public ResponseEntity<Void> deleteOrder(@RequestParam(value = "id") Long id) {
            try {
                OrderEntity entity = orderService.queryOrderById(id);
                if (null == entity) {
                    // 不存在返回404
                    return ResponseEntity.status(HttpStatus.NOT_FOUND).build();
                }
                orderService.deleteOrderById(id);
                // 204
                return ResponseEntity.status(HttpStatus.NO_CONTENT).build();
            } catch (Exception e) {
                e.printStackTrace();
            }
            // 500
            return ResponseEntity.status(HttpStatus.INTERNAL_SERVER_ERROR).body(null);
        }
    }

四、 HTTP状态码说明
状态码     状态码英文名称     中文描述
1开头的状态码
100     Continue     继续。客户端应继续其请求
101     Switching Protocols     切换协议。服务器根据客户端的请求切换协议。只能切换到更高级的协议,例如,切换到HTTP的新版本协议
2开头的状态码
200     OK     请求成功。一般用于GET与POST请求
201     Created     已创建。成功请求并创建了新的资源
202     Accepted     已接受。已经接受请求,但未处理完成
203     Non-Authoritative Information     非授权信息。请求成功。但返回的meta信息不在原始的服务器,而是一个副本
204     No Content     无内容。服务器成功处理,但未返回内容。在未更新网页的情况下,可确保浏览器继续显示当前文档
205     Reset Content     重置内容。服务器处理成功,用户终端(例如:浏览器)应重置文档视图。可通过此返回码清除浏览器的表单域
206     Partial Content     部分内容。服务器成功处理了部分GET请求
3开头的状态码
300     Multiple Choices     多种选择。请求的资源可包括多个位置,相应可返回一个资源特征与地址的列表用于用户终端(例如:浏览器)选择
301     Moved Permanently     永久移动。请求的资源已被永久的移动到新URI,返回信息会包括新的URI,浏览器会自动定向到新URI。今后任何新的请求都应使用新的URI代替
302     Found     临时移动。与301类似。但资源只是临时被移动。客户端应继续使用原有URI
303     See Other     查看其它地址。与301类似。使用GET和POST请求查看
304     Not Modified     未修改。所请求的资源未修改,服务器返回此状态码时,不会返回任何资源。客户端通常会缓存访问过的资源,通过提供一个头信息指出客户端希望只返回在指定日期之后修改的资源
305     Use Proxy     使用代理。所请求的资源必须通过代理访问
306     Unused     已经被废弃的HTTP状态码
307     Temporary Redirect     临时重定向。与302类似。使用GET请求重定向
4开头的状态码
400     Bad Request     客户端请求的语法错误,服务器无法理解
401     Unauthorized     请求要求用户的身份认证
402     Payment Required     保留,将来使用
403     Forbidden     服务器理解请求客户端的请求,但是拒绝执行此请求
404     Not Found     服务器无法根据客户端的请求找到资源(网页)。通过此代码,网站设计人员可设置"您所请求的资源无法找到"的个性页面
405     Method Not Allowed     客户端请求中的方法被禁止
406     Not Acceptable     服务器无法根据客户端请求的内容特性完成请求
407     Proxy Authentication Required     请求要求代理的身份认证,与401类似,但请求者应当使用代理进行授权
408     Request Time-out     服务器等待客户端发送的请求时间过长,超时
409     Conflict     服务器完成客户端的PUT请求是可能返回此代码,服务器处理请求时发生了冲突
410     Gone     客户端请求的资源已经不存在。410不同于404,如果资源以前有现在被永久删除了可使用410代码,网站设计人员可通过301代码指定资源的新位置
411     Length Required     服务器无法处理客户端发送的不带Content-Length的请求信息
412     Precondition Failed     客户端请求信息的先决条件错误
413     Request Entity Too Large     由于请求的实体过大,服务器无法处理,因此拒绝请求。为防止客户端的连续请求,服务器可能会关闭连接。如果只是服务器暂时无法处理,则会包含一个Retry-After的响应信息
414     Request-URI Too Large     请求的URI过长(URI通常为网址),服务器无法处理
415     Unsupported Media Type     服务器无法处理请求附带的媒体格式
416     Requested range not satisfiable     客户端请求的范围无效
417     Expectation Failed     服务器无法满足Expect的请求头信息
5开头的状态码
500     Internal Server Error     服务器内部错误,无法完成请求
501     Not Implemented     服务器不支持请求的功能,无法完成请求
502     Bad Gateway     充当网关或代理的服务器,从远端服务器接收到了一个无效的请求
503     Service Unavailable     由于超载或系统维护,服务器暂时的无法处理客户端的请求。延时的长度可包含在服务器的Retry-After头信息中
504     Gateway Time-out     充当网关或代理的服务器,未及时从远端服务器获取请求
505     HTTP Version not supported     服务器不支持请求的HTTP协议的版本,无法完成处理

推荐文档:https://blog.igevin.info/posts/restful-architecture-in-general/
————————————————
版权声明:本文为CSDN博主「伊宇紫」的原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/LDY1016/article/details/85112631

通俗易懂的RESTful API实践详解(含代码)的更多相关文章

  1. Requests实践详解

    Requests是什么 Requests是用python语言基于urllib编写的,采用的是Apache2 Licensed开源协议的HTTP库 如果你看过上篇文章关于urllib库的使用,你会发现, ...

  2. ubuntu apache2配置详解(含虚拟主机配置方法)

    ubuntu apache2配置详解(含虚拟主机配置方法) 在Windows下,Apache的配置文件通常只有一个,就是httpd.conf.但我在Ubuntu Linux上用apt-get inst ...

  3. API函数详解:API大全总目录(按字母排列)

    API函数详解 http://www.feiesoft.com/api/api.html

  4. Hadoop生态圈-zookeeper的API用法详解

    Hadoop生态圈-zookeeper的API用法详解 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.测试前准备 1>.开启集群 [yinzhengjie@s101 ~] ...

  5. android Camera2 API使用详解

    原文:android Camera2 API使用详解 由于最近需要使用相机拍照等功能,鉴于老旧的相机API问题多多,而且新的设备都是基于安卓5.0以上的,于是本人决定研究一下安卓5.0新引入的Came ...

  6. Mercury:唯品会全链路应用监控系统解决方案详解(含PPT)

    Mercury:唯品会全链路应用监控系统解决方案详解(含PPT) 原创: 姚捷 高可用架构 2016-08-08    

  7. Android API Levels 详解

    Android API Levels 当你开发你的Android应用程序时,了解该平台API变更管理的基本方法和概念是很有帮助的.同样的,知道API级别标识以及该标识如何保障你的应用与实际硬件设备相兼 ...

  8. 深度学习之卷积神经网络(CNN)详解与代码实现(一)

    卷积神经网络(CNN)详解与代码实现 本文系作者原创,转载请注明出处:https://www.cnblogs.com/further-further-further/p/10430073.html 目 ...

  9. Python - 元组(tuple) 详解 及 代码

    元组(tuple) 详解 及 代码 本文地址: http://blog.csdn.net/caroline_wendy/article/details/17290967 元组是存放任意元素集合,不能修 ...

随机推荐

  1. 20190708三人开黑CF模拟赛

    7月8号晚上8点和两位巨佬开了一场虚拟cf: [Helvetic Coding Contest 2018 online mirror (teams allowed, unrated)] 我这么蔡,只A ...

  2. K8S部署遇到的问题处理汇总

    第一个: node节点注册提示:failed to get config map: Unauthorized 代码如下: [root@node1 ~]# kubeadm join --token ll ...

  3. 如何利用开源解码项目开发js视频解码的web应用 系列

    介绍web上开发视频业务相关程序的技术演变历程 https://www.cnblogs.com/maoliangwu/articles/12046495.html 介绍ffmpeg asm.js we ...

  4. Codeforces_734_E

    http://codeforces.com/problemset/problem/734/E 每次操作可以把连通的同颜色的点全部换颜色,缩点,找直径,第一遍dfs找起点,第二遍dfs求直径. #inc ...

  5. Go语言学习之goroutine

    协程Coroutine 特点 轻量级的"线程" 非抢占式多任务处理,由协程主动交出控制权 编译器/解释器/虚拟机层面的多任务,非操作系统 多个协程可以在一个或多个线程上执行 go关 ...

  6. PC微信逆向--实现消息防撤回

    自从聊天软件消息撤回功能问世后,对于撤回的消息,我们对它一直有种强烈的好奇感."Ta刚撤回了什么?是骂我的话?还是说喜欢我?还是把发给其他人的消息误发给了我?好气呀,都看不到了...&quo ...

  7. PyObject and PyTypeObject - Python 中的 '对象' 们

    1 PyObject, PyTypeObject - Python 中的 '对象' 们 '一切皆对象' - 这是 Python 的学习和使用者们最最常听到一句, 可谓 博大精深 - '勃大精深'. ' ...

  8. centos7 上为php-fpm安装gd扩展库

    转自:https://blog.csdn.net/liyyzz33/article/details/89166110 首先查看自己当前php的版本 php -v PHP 5.6.40 查看yum中是否 ...

  9. 一键安装apache-2.4.38脚本

    [root@lamp scripts]# cat /etc/redhat-release CentOS Linux release 7.5.1804 (Core) [root@lamp scripts ...

  10. Element-UI ( Dropdow )下拉菜单组件command传输对象

    通过 :command绑定对象数据,handleCommand方法处理数据 template <div v-for="(item, index) in FlyWarningList&q ...