1.RESTful規範
一 面向資源編程
每一個URL表明一種資源,URL中儘可能不要用動詞,要用名詞。
二 根據method不一樣,進行不一樣的操做
GET/POST/PUT/DELETE/PATCH
三 在URL中體現版本
https://www.bootcss.com/v1/mycss
https://v1.bootcss.com/mycss
四 在URL中體現是不是API
https://www.bootcss.com/api/mycss
https://api.bootcss.com/mycss
五 在URL中的過濾條件
https://www.bootcss.com/v1/mycss?page=3
六 儘可能使用HTTPS
https://www.bootcss.com/v1/mycss
七 響應時設置狀態碼
1** 信息,服務器收到請求,須要請求者繼續執行操做
2** 成功,操做被成功接收並處理
3** 重定向,須要進一步的操做以完成請求
4** 客戶端錯誤,請求包含語法錯誤或沒法完成請求
5** 服務器錯誤,服務器在處理請求的過程當中發生了錯誤
八 返回值
GET請求 返回查到全部或單條數據
POST請求 返回新增的數據
PUT請求 返回更新數據
PATCH請求 局部更新 返回更新整條數據
DELETE請求 返回值爲空
九 返回錯誤信息
返回值攜帶錯誤信息
十 Hypermedia API
若是遇到須要跳轉的狀況 攜帶跳轉接口的URL
ret = {
code: 1000,
data:{
id:1,
name:'小強',
depart_id:http://www.luffycity.com/api/v1/depart/8/
}
}
2.DRF序列化
2.1爲何要使用DRF來序列化
當咱們作先後端分離的項目~~咱們先後端交互通常都選擇JSON數據格式,JSON是一個輕量級的數據交互格式。
那麼咱們給前端數據的時候都要轉成json格式,那就須要對咱們從數據庫拿到的數據進行序列化。
接下來咱們看下django序列化和rest_framework序列化的對比~~
2.2Django的序列化
#1.Jsonresponse序列化方式
class BooksView(View):
def get(self, request):
book_list = Book.objects.values("id", "title", "chapter", "pub_time", "publisher")
book_list = list(book_list)
# 若是咱們須要取外鍵關聯的字段信息 須要循環獲取外鍵 再去數據庫查而後拼接成咱們想要的
ret = []
for book in book_list:
pub_dict = {}
pub_obj = Publish.objects.filter(pk=book["publisher"]).first()
pub_dict["id"] = pub_obj.pk
pub_dict["title"] = pub_obj.title
book["publisher"] = pub_dict
ret.append(book)
ret = json.dumps(book_list, ensure_ascii=False, cls=MyJson)
return HttpResponse(ret)
# json.JSONEncoder.default()
# 解決json不能序列化時間字段的問題
class MyJson(json.JSONEncoder):
def default(self, field):
if isinstance(field, datetime.datetime):
return field.strftime('%Y-%m-%d %H:%M:%S')
elif isinstance(field, datetime.date):
return field.strftime('%Y-%m-%d')
else:
return json.JSONEncoder.default(self, field)
#2.Django序列化方法
from django.core import serializers
# 可以獲得咱們要的效果 結構有點複雜
class BooksView(View):
def get(self, request):
book_list = Book.objects.all()
ret = serializers.serialize("json", book_list)
return HttpResponse(ret)
2.3DRF的序列化
2.3.1drf序列化
-- Django咱們CBV繼承類是View,如今DRF咱們要用APIView
-- Django中返回的時候咱們用HTTPResponse,JsonResponse,render ,DRF咱們用Response
from rest_framework import serializers
class BookSerializer(serializers.Serializer):
id = serializers.IntegerField()
title = serializers.CharField(max_length=32)
CHOICES = ((1, "Linux"), (2, "Django"), (3, "Python"))
# 這是爲了顯示choice字段的值
chapter = serializers.ChoiceField(choices=CHOICES, source="get_category_display")
pub_time = serializers.DateField()
from rest_framework.views import APIView
from rest_framework.response import Response
from app.models import Book, Publisher
from app.serializers import BookSerializer
class BookView(APIView):
def get(self, request):
book_list = Book.objects.all()
ret = BookSerializer(book_list, many=True)
return Response(ret.data)
2.3.2外鍵關係序列化
#serialisers.py
from django.db import models
# Create your models here.
__all__ = ["Book", "Publisher", "Author"]
class Book(models.Model):
title = models.CharField(max_length=32, verbose_name="圖書名稱")
CHOICES = ((1, "Python"), (2, "Go"), (3, "Linux"))
category = models.IntegerField(choices=CHOICES, verbose_name="圖書的類別")
pub_time = models.DateField(verbose_name="圖書的出版日期")
publisher = models.ForeignKey(to="Publisher", on_delete=None)
author = models.ManyToManyField(to="Author")
def __str__(self):
return self.title
class Meta:
verbose_name_plural = "book"
class Publisher(models.Model):
title = models.CharField(max_length=32, verbose_name="出版社的名稱")
def __str__(self):
return self.title
class Meta:
verbose_name_plural = "publish"
class Author(models.Model):
name = models.CharField(max_length=32, verbose_name="做者的姓名")
def __str__(self):
return self.name
class Meta:
verbose_name_plural = "author"
#views.py
from rest_framework.views import APIView
from rest_framework.response import Response
from app.models import Book, Publisher
from app.serializers import BookSerializer
class BookView(APIView):
def get(self, request):
book_list = Book.objects.all()
ret = BookSerializer(book_list, many=True)
return Response(ret.data)
2.3.3DRF反序列化
當前端給咱們發post的請求的時候~前端給咱們傳過來的數據~咱們要進行一些校驗而後保存到數據庫~
這些校驗以及保存工做,DRF的Serializer也給咱們提供了一些方法了~~
首先~咱們要寫反序列化用的一些字段~有些字段要跟序列化區分開~
Serializer提供了.is_valid() 和.save()方法~~
使用write_only=True,read_only=True區別序列化字段和反序列化字段
必須重寫create方法
from rest_framework import serializers
from .models import Book
class PublisherSerializer(serializers.Serializer):
id = serializers.IntegerField()
title = serializers.CharField(max_length=32)
class AuthorSerializer(serializers.Serializer):
id = serializers.IntegerField()
name = serializers.CharField(max_length=32)
book_obj = {
"title": "vita的使用教程",
"w_category": 1,
"pub_time": "2018-10-09",
"publisher_id": 1,
"author_list": [1, 2]
}
class BookSerializer(serializers.Serializer):
# 插入數據時id自動生成,不須要校驗
id = serializers.IntegerField(required=False)
title = serializers.CharField(max_length=32)
CHOICES = ((1, "Python"), (2, "Go"), (3, "Linux"))
# read_only=True只在序列化的時候使用
category = serializers.ChoiceField(choices=CHOICES, source="get_category_display", read_only=True)
pub_time = serializers.DateField()
# write_only=True反序列化的字段,只在反序列化的時候使用,即保存數據的時候使用
w_category = serializers.ChoiceField(choices=CHOICES, write_only=True)
# 只在反序列化的時候使用,插入數據的時候,只須要插入publisher_id就能夠了
publisher_id = serializers.IntegerField(write_only=True)
# 必定要加上read_only=True,不然在插入數據的時候,提示該字段必須添加
publisher = PublisherSerializer(read_only=True)
# 指明是多對多的外鍵關係
author = AuthorSerializer(many=True, read_only=True)
# 一樣,只在反序列化,保存數據的時候使用
author_list = serializers.ListField(write_only=True)
def create(self, validated_data):
"""
反序列化的時候,必定要寫create方法
:param validated_data:
:return:
"""
book = Book.objects.create(title=validated_data["title"], category=validated_data["w_category"],
pub_time=validated_data["pub_time"], publisher_id=validated_data["publisher_id"])
# 多對多的建立
book.author.add(*validated_data["author_list"])
return book
#views.py
from rest_framework.views import APIView
from rest_framework.response import Response
from app.models import Book, Publisher
from app.serializers import BookSerializer
class BookView(APIView):
def get(self, request):
book_list = Book.objects.all()
ret = BookSerializer(book_list, many=True)
return Response(ret.data)
def post(self, request):
"""
反序列化,保存數據
:param request:
:return:
"""
print(request.data)
serializer = BookSerializer(data=request.data)
if serializer.is_valid():
serializer.save()
return Response(serializer.data)
else:
return Response(serializer.errors)
2.3.5獲取單條數據
#urls.py
from django.contrib import admin
from django.urls import path,include
from app.views import BookView,BookEditView
urlpatterns = [
path('admin/', admin.site.urls),
path('list', BookView.as_view()),
path('retrieve/<int:id>', BookEditView.as_view()),
]
#views.py
class BookEditView(APIView):
def get(self, request, id):
"""
獲取單條數據的接口
:param request:
:param id:
:return:
"""
book_obj = Book.objects.filter(id=id).first()
ret = BookSerializer(book_obj)
return Response(ret.data)
2.3.6PUT請求的部分驗證
# views.py
class BookEditView(APIView):
def get(self, request, id):
"""
獲取單條數據的接口
:param request:
:param id:
:return:
"""
book_obj = Book.objects.filter(id=id).first()
ret = BookSerializer(book_obj)
return Response(ret.data)
def put(self, request, id):
"""
用戶提交Put請求
:param request:
:param id:
:return:
"""
book_obj = Book.objects.filter(id=id).first()
# partial=True容許部分校驗
serializer = BookSerializer(book_obj, data=request.data, partial=True)
if serializer.is_valid():
serializer.save()
return Response(serializer.data)
else:
return Response(serializer.errors)
from rest_framework import serializers
from .models import Book
class PublisherSerializer(serializers.Serializer):
id = serializers.IntegerField()
title = serializers.CharField(max_length=32)
class AuthorSerializer(serializers.Serializer):
id = serializers.IntegerField()
name = serializers.CharField(max_length=32)
book_obj = {
"title": "vita的使用教程",
"w_category": 1,
"pub_time": "2018-10-09",
"publisher_id": 1,
"author_list": [1, 2]
}
put_data = {
"title": "vita的使用教程2"
}
class BookSerializer(serializers.Serializer):
# 插入數據時id自動生成,不須要校驗
id = serializers.IntegerField(required=False)
title = serializers.CharField(max_length=32)
CHOICES = ((1, "Python"), (2, "Go"), (3, "Linux"))
# read_only=True只在序列化的時候使用
category = serializers.ChoiceField(choices=CHOICES, source="get_category_display", read_only=True)
pub_time = serializers.DateField()
# write_only=True反序列化的字段,只在反序列化的時候使用,即保存數據的時候使用
w_category = serializers.ChoiceField(choices=CHOICES, write_only=True)
# 只在反序列化的時候使用,插入數據的時候,只須要插入publisher_id就能夠了
publisher_id = serializers.IntegerField(write_only=True)
# 必定要加上read_only=True,不然在插入數據的時候,提示該字段必須添加
publisher = PublisherSerializer(read_only=True)
# 指明是多對多的外鍵關係
author = AuthorSerializer(many=True, read_only=True)
# 一樣,只在反序列化,保存數據的時候使用
author_list = serializers.ListField(write_only=True)
def create(self, validated_data):
"""
反序列化的時候,必定要寫create方法
:param validated_data:
:return:
"""
book = Book.objects.create(title=validated_data["title"], category=validated_data["w_category"],
pub_time=validated_data["pub_time"], publisher_id=validated_data["publisher_id"])
# 多對多的建立
book.author.add(*validated_data["author_list"])
return book
def update(self, instance, validated_data):
"""
Put請求的方法
:param instance:
:param validated_data:
:return:
"""
#若是該字段沒有,就使用原來的數據
instance.title = validated_data.get("title", instance.title)
instance.category = validated_data.get("category", instance.category)
instance.pub_time = validated_data.get("pub_time", instance.pub_time)
instance.publisher_id = validated_data.get("publisher_id", instance.publisher_id)
# 多對多的處理,若是用戶傳了該字段,就更新
if validated_data.get("author_list"):
instance.author.set(validated_data["author_list"])
instance.save()
return instance
2.3.7DRF驗證
def my_validate(value):
"""
全局驗證
:param value:
:return:
"""
if "敏感信息" in value.lower():
raise serializers.ValidationError("不能含有敏感信息")
else:
return value
class BookSerializer(serializers.Serializer):
# 插入數據時id自動生成,不須要校驗
id = serializers.IntegerField(required=False)
title = serializers.CharField(max_length=32, validators=[my_validate])
CHOICES = ((1, "Python"), (2, "Go"), (3, "Linux"))
# read_only=True只在序列化的時候使用
category = serializers.ChoiceField(choices=CHOICES, source="get_category_display", read_only=True)
pub_time = serializers.DateField()
# write_only=True反序列化的字段,只在反序列化的時候使用,即保存數據的時候使用
w_category = serializers.ChoiceField(choices=CHOICES, write_only=True)
# 只在反序列化的時候使用,插入數據的時候,只須要插入publisher_id就能夠了
publisher_id = serializers.IntegerField(write_only=True)
# 必定要加上read_only=True,不然在插入數據的時候,提示該字段必須添加
publisher = PublisherSerializer(read_only=True)
# 指明是多對多的外鍵關係
author = AuthorSerializer(many=True, read_only=True)
# 一樣,只在反序列化,保存數據的時候使用
author_list = serializers.ListField(write_only=True)
def create(self, validated_data):
"""
反序列化的時候,必定要寫create方法
:param validated_data:
:return:
"""
book = Book.objects.create(title=validated_data["title"], category=validated_data["w_category"],
pub_time=validated_data["pub_time"], publisher_id=validated_data["publisher_id"])
# 多對多的建立
book.author.add(*validated_data["author_list"])
return book
def update(self, instance, validated_data):
"""
Put請求的方法
:param instance:
:param validated_data:
:return:
"""
#若是該字段沒有,就使用原來的數據
instance.title = validated_data.get("title", instance.title)
instance.category = validated_data.get("category", instance.category)
instance.pub_time = validated_data.get("pub_time", instance.pub_time)
instance.publisher_id = validated_data.get("publisher_id", instance.publisher_id)
# 多對多的處理,若是用戶傳了該字段,就更新
if validated_data.get("author_list"):
instance.author.set(validated_data["author_list"])
instance.save()
return instance
def validate_title(self, value):
"""
單個字段的驗證
:param value:
:return:
"""
if "python" not in value.lower():
raise serializers.ValidationError("標題必須含有python")
return value
def validate(self, attrs):
"""
多個字段的驗證
:param attrs:
:return:
"""
if attrs["w_category"] == 1 and attrs["publisher_id"] == 1:
return attrs
else:
raise serializers.ValidationError("分類以及標題不符合要求")
2.3.8 ModelSerializer
如今咱們已經清楚了Serializer的用法,會發現咱們全部的序列化跟咱們的模型都緊密相關~
那麼,DRF也給咱們提供了跟模型緊密相關的序列化器~~ModelSerializer~~
-- 它會根據模型自動生成一組字段
-- 它簡單的默認實現了.update()以及.create()方法
2.3.8.1定義一個ModelSerializer序列化器
class BookSerializer(serializers.ModelSerializer):
class Meta:
model = Book
fields = "__all__"
# fields = ["id", "title", "pub_time"]
# exclude = ["user"]
# 分別是全部字段 包含某些字段 排除某些字段
2.3.8.2外鍵關係的序列化
class BookSerializer(serializers.ModelSerializer):
class Meta:
model = Book
fields = "__all__"
# fields = ["id", "title", "pub_time"]
# exclude = ["user"]
# 分別是全部字段 包含某些字段 排除某些字段
depth = 1
# depth 表明找嵌套關係的第幾層
2.3.8.3自定義字段
咱們能夠聲明一些字段來覆蓋默認字段,來進行自定製~
好比咱們的選擇字段,默認顯示的是選擇的key,咱們要給用戶展現的是value。
class BookSerializer(serializers.ModelSerializer):
chapter = serializers.CharField(source="get_chapter_display", read_only=True)
class Meta:
model = Book
fields = "__all__"
# fields = ["id", "title", "pub_time"]
# exclude = ["user"]
# 分別是全部字段 包含某些字段 排除某些字段
depth = 1
2.3.8.4meta中其餘關鍵字參數
class BookSerializer(serializers.ModelSerializer):
chapter = serializers.CharField(source="get_chapter_display", read_only=True)
class Meta:
model = Book
fields = "__all__"
# fields = ["id", "title", "pub_time"]
# exclude = ["user"]
# 分別是全部字段 包含某些字段 排除某些字段
depth = 1
read_only_fields = ["id"]
extra_kwargs = {"title": {"validators": [my_validate,]}}
2.3.8.5post以及patch請求
因爲depth會讓咱們外鍵變成只讀,因此咱們再定義一個序列化的類,其實只要去掉depth就能夠了~~
class BookSerializer(serializers.ModelSerializer):
chapter = serializers.CharField(source="get_chapter_display", read_only=True)
class Meta:
model = Book
fields = "__all__"
# fields = ["id", "title", "pub_time"]
# exclude = ["user"]
# 分別是全部字段 包含某些字段 排除某些字段
read_only_fields = ["id"]
extra_kwargs = {"title": {"validators": [my_validate,]}}
2.3.8.6SerializerMethodField
外鍵關聯的對象有不少字段咱們是用不到的~都傳給前端會有數據冗餘~就須要咱們本身去定製序列化外鍵對象的哪些字段~~
class BookSerializer(serializers.ModelSerializer):
chapter = serializers.CharField(source="get_chapter_display", read_only=True)
user = serializers.SerializerMethodField()
publisher = serializers.SerializerMethodField()
def get_user(self, obj):
# obj是當前序列化的book對象
users_query_set = obj.user.all()
return [{"id": user_obj.pk, "name": user_obj.name} for user_obj in users_query_set]
def get_publisher(self, obj):
publisher_obj = obj.publisher
return {"id": publisher_obj.pk, "title": publisher_obj.title}
class Meta:
model = Book
fields = "__all__"
# fields = ["id", "title", "pub_time"]
# exclude = ["user"]
# 分別是全部字段 包含某些字段 排除某些字段
read_only_fields = ["id"]
extra_kwargs = {"title": {"validators": [my_validate,]}
2.3.8.7用ModelSerializer改進上面Serializer的完整版
class BookSerializer(serializers.ModelSerializer):
dis_chapter = serializers.SerializerMethodField(read_only=True)
users = serializers.SerializerMethodField(read_only=True)
publishers = serializers.SerializerMethodField(read_only=True)
def get_users(self, obj):
# obj是當前序列化的book對象
users_query_set = obj.user.all()
return [{"id": user_obj.pk, "name": user_obj.name} for user_obj in users_query_set]
def get_publishers(self, obj):
publisher_obj = obj.publisher
return {"id": publisher_obj.pk, "title": publisher_obj.title}
def get_dis_chapter(self, obj):
return obj.get_chapter_display()
class Meta:
model = Book
# fields = "__all__"
# 字段是有序的
fields = ["id", "title","dis_chapter", "pub_time", "publishers", "users","chapter", "user", "publisher"]
# exclude = ["user"]
# 分別是全部字段 包含某些字段 排除某些字段
read_only_fields = ["id", "dis_chapter", "users", "publishers"]
extra_kwargs = {"title": {"validators": [my_validate,]}, "user": {"write_only": True}, "publisher": {"write_only": True},
"chapter": {"write_only": True}}