当前位置: 首页 > news >正文

restful api和普通api有什么特点_Django REST Framework教程(1): 为什么要学习DRF, 什么是序列化和RESTful的API...

小编曾经承诺的Django REST Framework原创系列教程终于赶上了2020年那年夏天的尾巴,姗姗而来。我们将用不少于10篇的原创文章系统性地介绍如何使用这个框架开发符合RESTful规范的API,并详细介绍API开发和管理中的核心知识比如关联模型的序列化、分页、过滤、认证、权限和限流。如果你已有一定Django基础知识并希望在API开发和前后端分离开发技术上更近一步, 请欢迎关注我们的微信公众号【Python Web与Django开发】并加星标哦。

f6a48202e75b3cf38de145b8dcd11fbf.png

为什么要学习Django REST  Framework?

Django的未来与Web开发未来发展趋势紧密相关。Django这种基于MVC开发模式的传统框架,非常适合开发基于PC的传统网站,因为它同时包括了后端的开发(逻辑层,数据库层) 和前端的开发(如模板语言,样式)。

然而最近几年及未来几年更流行的开发模式肯定是前后端分离, Django也需要做出自己的改变来适应这种开发模式。现代网络应用Web APP或大型网站一般是一个后台,然后对应各种客户端(iOS, android, 浏览器)。由于客户端的开发语言与后台的开发语言经常不一样,这时需要后台能够提供可以跨平台跨语言的一种标准的资源或数据(如json格式)供前后端沟通,这就是Web API(网络应用程序接口)的作用了。

Django本身并不是为了开发符合REST规范的Web API而设计, 不过借助Django REST Framework (DRF)我们可以快速开发出优秀规范的Web API来。所以我们这里要感谢DRF,因为它,Django的应用前景更广了,减少了被淘汰的风险。

e16e660839ae718ecdd4f46e1e5a5353.png

Django REST Framework的介绍及安装

Django REST framework (DRF)是基于Django实现的一个RESTful风格API框架,能够帮助我们快速开发RESTful风格的API,文档地址如下所示:

  • 官网:https://www.django-rest-framework.org/

  • 中文文档:https://q1mi.github.io/Django-REST-framework-documentation/

DRF可以使用pip安装,安装前请确保你已经安装了Django。

pip install djangorestframework

如果想要获取一个图形化的页面来操作API,需要将 rest_framework 注册到项目的INSTALL_APPS中,如下所示:

INSTALLED_APPS = [    'django.contrib.admin',    'django.contrib.auth',    'django.contrib.contenttypes',    'django.contrib.sessions',    'django.contrib.messages',    'django.contrib.staticfiles',    'rest_framework',    'your_app', # 你自己的app]

现在我们已经安装好DRF了,但在正式使用它开发API前我们还需要了解两件事情。什么是数据的序列化(Serialization)以及什么是RESTful规范的API, 这对于理解DRF中的serializers类和每个API端点对应的url设计至关重要。

Python类型数据序列化(serialization)

每种编程语言都有各自的数据类型, 将属于自己语言的数据类型或对象转换为可通过网络传输或可以存储到本地磁盘的数据格式(如:XML、JSON或特定格式的字节串)的过程称为序列化(seralization);反之则称为反序列化。API开发的本质就是各种后端语言的自己的数据类型序列化为通用的可读可传输的数据格式,比如常见的JSON类型数据。

举个简单例子。Python自带json模块的dumps方法可以将python常用数据格式(比如列表或字典)转化为json格式,如下所示。你注意到了吗? 生成的json格式数据外面都加了单引号,这说明dict类型数据已经转化成了json字符串。

>>> import json>>> json.dumps({"name":"John", "score": 112})'{"name": "John", "score": 112}'

Django编程就是是python编程,以上所介绍的序列化方法对django也是适用的。不同的是Django还有自己专属的数据类型比如查询集QuerySet和ValueQuerySet类型数据,还提供了更便捷的serializers类。使用Django自带的serializers类也可以轻易将QuerySet格式的数据转化为json格式。

# Django Queryset数据 to Jsonfrom django.core import serializersdata = serializers.serialize("json", SomeModel.objects.all())data1 = serializers.serialize("json", SomeModel.objects.all(), fields=('name','id'))data2 = serializers.serialize("json", SomeModel.objects.filter(field = some_value))

有时候我们只需要查询结果集的部分字段,可以使用values('字段名','字段名2')来要求返回的是哪些列的数据.但是返回来的是ValueQuerySet对象而不是QuerySet对象。ValuesQuerySet对象不能用 serializers.serialize() 方法序列化成json, 需要先转换成list再用 json.dumps()方法序列化成json格式。示例代码如下所示:

import jsonfrom django.core.serializers.json import DjangoJSONEncoderqueryset = myModel.objects.filter(foo_icontains=bar).values('f1', 'f2', 'f3')data4 = json.dumps(list(queryset), cls=DjangoJSONEncoder)

尽管Django自带的serializer类也能将Django的查询集QuerySet序列化成json格式数据,Django REST Framework才是你真正需要的序列化工具。与django自带的serializers类相比,rest framework支持token认证、过滤和限流等多种强大功能,我们后面会陆续详细介绍。

什么是符合RESTful规范的API?

REST是REpresentational State Transfer三个单词的缩写,由Roy Fielding于2000年论文中提出。简单来说,就是用URI表示资源,用HTTP方法(GET, POST, PUT, DELETE)表征对这些资源进行操作。而如果想你的api被称为restful api,只要遵循其规定的约束。网上有很多文章对RESTful API规范做了详细介绍, 比如阮一峰的博客和简书, 我们这里只摘取总结部分精华内容,因为我们后面会用到。

协议、域名和版本

尽量使用https协议,使用专属域名来提供API服务,并在URL里标注api版本,如下所示

https://api.example.com/v1

https://www.example.com/api/v1 

uri(统一资源标识符)

在RESTful架构中,每个网址代表一种资源(resource),这个网络地址就是uri (uniform resource identifier), 有时也被称为URL(uniform resource locator)。 因为URI对应一种资源,所以里面不能有动词,只能有名词。一般来说,数据库中的表都是同种记录的"集合"(collection),所以API中的名词也应该使用复数。

https://api.example.com/v1/users # 用户列表资源地址

https://api.example.com/v1/users/{id}  # 用户id=5资源。注意:这里是users/5,而不是user/5

https://api.example.com/v1/users/{id}/articles  # 用户id=5发表的文章列表

如果需要对一个用户信息进行编辑或删除,一个传统Django开发者可能将URL写成如下所示

https://api.example.com/v1/users/{id}/edit/ # 编辑用户

https://api.example.com/v1/users/{id}/delete/  # 删除用户

上面URL设计其实是不符合RESTful规范的。一个 URI就应该是一个资源,本身不能包含任何动作。REST的规范是资源的URI地址是固定不变的,对同一资源应使用不同的HTTP请求方法进行不同的操作,比如常见的增删查改。

[POST]    https://api.example.com/v1/users   // 新增[GET]     https://api.example.com/v1/users/1 // 查询[PATCH]   https://api.example.com/v1/users/1 // 更新[PUT]     https://api.example.com/v1/users/1 // 覆盖,全部更新[DELETE]  https://api.example.com/v1/users/1 // 删除

有时候URL比较长,可能由多个单词组成,建议使用中划线"-"分割,而不是下划线"_"作为分隔符,另外每个url的结尾不能加斜线"/"。

https://api.example.com/v1/user-management/users/{id} # 好

https://api.example.com/v1/user_management/users/{id}  # 不好

https://api.example.com/v1/user-management/users/{id}/  # 不好

HTTP请求方法

常用的HTTP请求方法有下面五个(括号里是对应的SQL命令),每个方法对应一个操作。客户端在消费服务器提供的API服务时不仅要指定请求地址,还需指定请求方法。

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

另外还有两个不常用方法HEADOPTIONSHEAD请求的是资源的元数据,比如一张照片,的元数据则可能包含了,照片拍摄的设备,地点,时间等。服务器一般将对应资源的元数据置于响应的报头集合返回给客户端,这样的响应一般不具有主体部分。OPTIONS则是发送一种“探测”请求以确定针对某个目标地址的请求必须具有怎样的约束(比如应该采用怎样的HTTP方法以及自定义的请求报头),然后根据其约束发送真正的请求。

小编注:不像Django,DRF支持所有以上请求方法。

过滤信息(Filtering)

如果记录数量很多,服务器不可能都将它们返回给用户。符合RESTful规范的API应该支持过滤。下面是一些常见的过滤参数。

?limit=10:指定返回记录的数量?offset=10:指定返回记录的开始位置。?page=2&per_page=100:指定第几页,以及每页的记录数。?sortby=name&order=asc:指定返回结果按照哪个姓名排序,以及排序顺序。?user_type_id=1:指定筛选条件,比如用户类型

小编注:DRF与django-filter联用可以轻松实现过滤。

状态码(Status Codes)

服务器在处理客户端请求后还应向用户返回响应的状态码和提示信息,常见的有以下一些状态码。

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 - [*]:服务器发生错误

小编注:DRF给出Respone时可以指定各种各样状态码,很容易使用。

Hypermedia API

RESTful API最好做到Hypermedia,即返回结果中提供链接,连向其他API方法,使得用户不查文档,也知道下一步应该做什么。比如,当用户向api.example.com的根目录发出请求,会得到这样一个文档。

{"link": {  "rel":   "collection https://www.example.com/zoos",  "href":  "https://api.example.com/zoos",  "title": "List of zoos",  "type":  "application/vnd.yourformat+json"}}

小编注:DRF支持HyperLinkedModel,很容易实现这点。

参考:http://www.ruanyifeng.com/blog/2014/05/restful_api.html

小结

本文总结了为什么要学习Django REST Framework, 什么是序列化和符合RESTful规范的API。下文中我们将以一个简单的博客的实例来详细介绍DRF的使用,开发最基础的API,并测试API的使用,欢迎关注。

大江狗

2020.08

相关阅读

Django基础连载系列-大江狗

Django的未来在哪里?

一文看懂Python及Django不同类型数据的json序列化(面试工作必读)

e72b6f1b50b8299bbfa5a3cf4e3e8484.png

相关文章:

  • 北京网站建设多少钱?
  • 辽宁网页制作哪家好_网站建设
  • 高端品牌网站建设_汉中网站制作
  • sql语言编程学习_在读大学生是否该重点学习一门编程语言,以及该如何选择编程语言...
  • .net 反编译_.net反编译的相关问题
  • class括号里的object_Class文件结构全面解析(下)
  • 标签打印模板_海鸟贴纸打印机:工作和生活中的得力助手!专治强迫症
  • gitlab run成功 但无法访问_用 GitLab 做 CI/CD 是什么感觉,太强了!!
  • 当前操作系统缺少黑体等字体_OpenBSD6.6正式版发布:多平台,类Unix的最安全操作系统...
  • wps 模拟分析 规划求解_基于长时间尺度的园区综合能源系统随机规划
  • 纬地8.0支持的cad版本_智慧社区弱电工程设计图纸,可编辑(CAD版本)弱电新人学习!...
  • markdown 公式_还在用MathType编写数学公式吗,markdown纯手撸复杂数学公式
  • tipi 深入理解php内核 pdf_深入理解 PHP7 内核之 HashTable
  • ps cs6 磨皮插件_Portraiture 3 for mac(ps磨皮滤镜插件) v3.5.4(3540)版
  • python index函数是左闭右开吗_Python容器类型公共方法总结
  • flutter 图表_Flutter 与 Chrome OS 珠联璧合
  • 计算加减乘除混合运算python实现_python,实现计算器程序,加减乘除混合运算加括号,完善实现...
  • 页面布局让footer居页面底部_网站页面底部固定的方法
  • JS 中的深拷贝与浅拷贝
  • 【comparator, comparable】小总结
  • Android 控件背景颜色处理
  • CentOS从零开始部署Nodejs项目
  • java取消线程实例
  • python大佬养成计划----difflib模块
  • vue2.0一起在懵逼的海洋里越陷越深(四)
  • Wamp集成环境 添加PHP的新版本
  • 第2章 网络文档
  • 如何用vue打造一个移动端音乐播放器
  • 少走弯路,给Java 1~5 年程序员的建议
  • 数组大概知多少
  • 我这样减少了26.5M Java内存!
  • 用mpvue开发微信小程序
  • FaaS 的简单实践
  • ​LeetCode解法汇总2304. 网格中的最小路径代价
  • # 透过事物看本质的能力怎么培养?
  • #{}和${}的区别是什么 -- java面试
  • #1015 : KMP算法
  • #HarmonyOS:软件安装window和mac预览Hello World
  • (2)STL算法之元素计数
  • (55)MOS管专题--->(10)MOS管的封装
  • (C#)一个最简单的链表类
  • (Matalb时序预测)WOA-BP鲸鱼算法优化BP神经网络的多维时序回归预测
  • (备份) esp32 GPIO
  • (六)DockerCompose安装与配置
  • (三)Pytorch快速搭建卷积神经网络模型实现手写数字识别(代码+详细注解)
  • (幽默漫画)有个程序员老公,是怎样的体验?
  • *2 echo、printf、mkdir命令的应用
  • .NET中 MVC 工厂模式浅析
  • .pyc文件还原.py文件_Python什么情况下会生成pyc文件?
  • .skip() 和 .only() 的使用
  • /bin/bash^M: bad interpreter: No such file ordirectory
  • @DependsOn:解析 Spring 中的依赖关系之艺术
  • @Import注解详解
  • @JoinTable会自动删除关联表的数据
  • @Mapper作用
  • @Transient注解
  • [012-1].第12节:Mysql的配置文件的使用
  • [AIGC] Nacos:一个简单 yet powerful 的配置中心和服务注册中心