rest-framework框架之序列化

开发 API 接口最重要的工作就是将代码片段的输出序列化和反序列化为类似于 json 的表示形式。

在 rest-framework 中,通过声明与 Django 中 forms 非常类似的序列化器(serializers)来实现。

用到的 models

from django.db import models

# Create your models here.

class Book(models.Model):
title = models.CharField(max_length=32)
price = models.IntegerField()
pub_date = models.DateField()
publish = models.ForeignKey("Publish")
authors = models.ManyToManyField("Author") def __str__(self):
return self.title class Publish(models.Model):
name = models.CharField(max_length=32)
email = models.EmailField() def __str__(self):
return self.name class Author(models.Model):
name = models.CharField(max_length=32)
age = models.IntegerField() def __str__(self):
return self.name

使用 Serializer

Serializer 的学习可以类比 Django 中的 form 。它们都有 CharField ,IntegerField 等。

数据表中的普通字段用 CharField ,IntegerField 等,一对多用 CharField ,里面加上 source 指向要对应的字段。多对多用 SerializerMethodField ,并且自定义一个名为 get_多对多字段名 的函数,此时该多对多字段的值取决于这个函数的返回值。

from app01.models import *
from rest_framework import serializers class BookSerializers(serializers.Serializer):
title = serializers.CharField(max_length=32)
price = serializers.IntegerField()
pub_date = serializers.DateField()
# 一对多用CharField,里面加上source
publish = serializers.CharField(source="publish.name")
# 多对多用SerializerMethodField
authors = serializers.SerializerMethodField() # 此时authors值取决于下面函数的返回值
def get_authors(self, obj):
authors_temp = []
for authors_obj in obj.authors.all():
authors_temp.append(authors_obj.name) return ",".join(authors_temp)

使用 ModelSerializer

ModelSerializer 可以类比成 Django 中的 ModelForm 它实现的功能与 Serializer 一样,只是代码量更少,功能更多。

普通字段

通过 Meta 类将普通字段转成 Serializer 类中的各种字段。其中, model 绑定的数据表模型,fields 绑定要转化的字段,"__all__" 表示要将所有字段都转成相应的类型。

class PublishModelSerializers(serializers.ModelSerializer):
class Meta:
model = Publish
fields = "__all__"

特殊字段

特殊字段可以自己添加,和 Serializer 一样,一对多字段用 CharField,里面加上 source ,多对多用 SerializerMethodField ,特殊字段依然可以用 get_字段名 来返回该字段的值。

注意:存在定制字段,需要改写 create 方法。

class BookModelSerializers(serializers.ModelSerializer):
# 默认将普通字段转化
class Meta:
model = Book
fields = "__all__" # 特殊字段可以自己添加
# 一对多用CharField,里面加上source
publish = serializers.CharField(source="publish.name")
publish = serializers.CharField(source="publish.pk")
# 多对多用SerializerMethodField
authors = serializers.SerializerMethodField() # 此时authors值取决于下面函数的返回值
def get_authors(self, obj):
authors_temp = []
for authors_obj in obj.authors.all():
authors_temp.append(authors_obj.name) return ",".join(authors_temp) # 存在定制字段,所以要改写create方法
# def create(self, validated_data):
# print("validated_data", validated_data)
# book_obj = Book.objects.create(title=validated_data["title"], price=validated_data["price"], pub_date=validated_data["pub_date"], publish_id=validated_data["publish"]["pk"])
# book_obj.authors.add(*validated_data["authors"])
#
# return book_obj

视图部分

需要用到 rest-framework 中的 APIView 类。

对每个数据表模型建立相应的视图类,并在视图类中通过不同的方法来接受前端以不同方法发来的数据。

from rest_framework.views import APIView
from .serializer import * # Create your views here.
class BookViewSet(APIView):
def get(self, request, *args, **kwargs):
... def post(self, request, *args, **kwargs):
... def put(self, request, *args, **kwargs):
... def patch(self, request, *args, **kwargs):
...

序列化方法一

使用 Django 中的 model_to_dict 。

book_list = Book.objects.all()
from django.forms.models import model_to_dict
import json
data = []
for obj in book_list:
data.append(model_to_dict(obj))
return HttpResponse(json.dumps(data))

序列化方法二

使用 Serializer 将数据转成 json 数据。

data = serializers.serialize("json", book_list)
return HttpResponse(data)

序列化方法三

直接通过 Serializer 或者 ModelSerializer 来返回。

bs = BookSerializers(book_list, many=True)
bs = BookModelSerializers(book_list, many=True) # 超链接时需要增加参数
bs = BookModelSerializers(book_list, many=True, context={'request': request})
return Response(bs.data)

提交 post 请求

与 Django 中的 ModelForm 类似,ModelSerializer 也有 is_valid 函数来判断验证是否通过, save 函数来讲数据保存到数据库中。

    def post(self, request, *args, **kwargs):
bs = BookSerializers(data=request.data, many=False)
request.data:post的数据
bs = BookModelSerializers(data=request.data, many=False) # 超链接时需要增加参数
bs = BookModelSerializers(data=request.data, many=False, context={'request': request}) if bs.is_valid():
bs.save()
return Response(bs.data)
else:
return HttpResponse(bs.errors)

单条数据的请求

单条数据与之前的区别在于多传过来一个单条数据的 ID ,通过该 ID 值进行筛选工作。

class PublishViewSet(APIView):

    def get(self, request, *args, **kwargs):
publish_list = Publish.objects.all()
ps = PublishModelSerializers(publish_list, many=True) return Response(ps.data) def post(self, request):
ps = PublishModelSerializers(data=request.data, many=False) if ps.is_valid():
print(ps.validated_data)
ps.save()
return Response(ps.data)
else:
return Response(ps.error)

超链接 API

有时,想让该字段的输出值为超链接,这样也是可以实现的。需要将该字段设置为

HyperlinkedIdentityField 类型,其中 view_name 为在 urls.py 中设置的视图名, lookup_field 绑定数据表模型的 ID , lookup_url_kwarg 绑定为主键。

class BookModelSerializers(serializers.ModelSerializer):
# 默认将普通字段转化
class Meta:
model = Book
fields = "__all__" # 定制url
publish = serializers.HyperlinkedIdentityField(
view_name="detailpublish",
lookup_field="publish_id",
lookup_url_kwarg="pk"
)

而包含超链接的字段进行实例化时,需要添加相应的参数:

bs = BookModelSerializers(book_list, many=True, context={'request': request})

urls 部分

对于视图类,只需要在 URL 中调用 as_view 函数即可。

注意:

  1. 整体数据和单条数据的操作不能用同一个视图类。
  2. 如果有超链接,需要给每个 URL 都加上 name 属性。
from django.conf.urls import url
from django.contrib import admin
from app01 import views urlpatterns = [
url(r'^admin/', admin.site.urls),
url(r'^books/$', views.BookViewSet.as_view(), name="book"),
url(r'^publishes/$', views.PublishViewSet.as_view(), name="publish"),
url(r'^publishes/(?P<pk>\d+)/$', views.PublishDetailView.as_view(), name="detailpublish"),
url(r'^authors/$', views.AuthorViewSet.as_view(), name="author"),
url(r'^books/(\d+)/$', views.BookDetailView.as_view(), name="detailbook"),
]

rest-framework框架之序列化的更多相关文章

  1. python-django rest framework框架之序列化

    序列化与反序列化: 对象 -> 字符串 序列化 字符串 -> 对象 反序列化 rest framework序列化+Form 目的: 解决QuerySet序列化问题 功能:解析 和 过滤 - ...

  2. 基于Django的Rest Framework框架的序列化组件

    本文目录 一 Django自带序列化组件 二 rest-framework序列化之Serializer 三 rest-framework序列化之ModelSerializer 四 生成hypermed ...

  3. Django_rest framework 框架介绍

    restful介绍  restful协议 一切皆是资源,操作只是请求方式 URL 设计 原先的URL设计方式 在url 中体现出操作行为 /books/ books /books/add/ addbo ...

  4. Django之Rest Framework框架

    一.什么是RESTful REST与技术无关,代表的是一种软件架构风格,REST是Representational State Transfer的简称,中文翻译为“表征状态转移” REST从资源的角度 ...

  5. Django----Rest Framework框架

    Django Rest Framework框架(10) - RESTful规范 1.API与用户的通信协议,总是使用HTTPs协议. 2.域名 https://api.example.com 尽量将A ...

  6. web前端Vue+Django rest framework 框架 生鲜电商项目实战视频教程 ☝☝☝

    web前端Vue+Django rest framework 框架 生鲜电商项目实战视频教程    web前端Vue+Django rest framework 框架 生鲜电商项目实战视频教程 学习 ...

  7. DRF框架之序列化器初体验

    首先,我们需要明白序列化和反序列化的过程指的是什么. 序列化操作:将模型数据 ---> 字典数据 --->JSON数据(响应JSON数据的操作) 反序列化操作:将JSON数据 ---> ...

  8. .Net 初步学习笔记之一——.Net 平台与.Net FrameWork框架的关系

    .Net 包含两部分 .Net平台 和.Net FrameWork 框架 1..Net FrameWork框架包含于.Net平台. .Net FrameWork提供环境和支撑保证.Net平台运行. 2 ...

  9. Swift - 访问通讯录-使用AddressBook.framework和AddressBookUI.framework框架实现

    1,通讯录访问介绍 通讯录(或叫地址簿,电话簿)是一个数据库,里面储存了联系人的相关信息.要实现访问通讯录有如下两种方式: (1)AddressBook.framework框架 : 没有界面,通过代码 ...

  10. Play Framework框架 JPA惯用注解

    Play Framework框架 JPA常用注解 1.@Entity(name=”EntityName”) 必须 ,name 为可选 , 对应数据库中一的个表 2.@Table(name=”" ...

随机推荐

  1. c++ 适配器模式(adapter)

    当两个系统的接口不一样时,我们就要重新封装一下接口,以便于当前系统的调用.这种模式叫做适配器模式.适配器模式分为两种: 1.对象组合适配器(Object Adapter) 2.类适配器(Class A ...

  2. SpringBoot26 RestTemplate、WebClient

    1 RestTemplate RestTemplate是在客户端访问 Restful 服务的一个核心类:RestTemplate通过提供回调方法和允许配置信息转换器来实现个性化定制RestTempla ...

  3. Shell +Cygwinterminal+WinMySQL 传参数授权

    前言:新公司因为部分业务原因有好几百组win机器装MySQL授权登录比较麻烦,简单的写了一个shell传值自动授权的脚本,保存复用. #!/bin/bash #author liding@zlhy.c ...

  4. 2014年Linux 和开源技术回顾盘点

    ZDNet科技观察家StevenJ.Vaughan-Nichols在年终发表了对Linux和开源技术这一年跌宕起伏的总结,细数这一年中的惊喜和不堪. 2014Linux之殇 “心脏出血(Heartbl ...

  5. CSS--抽屉(dig.chouti.com)页面

    一.设置整体页面宽度 一般写个样式命名为.d{}设置整体页面指定宽度和居中,京东命名为.w{},bootstrap里命名为.container{} 1 2 3 4 5 6 7 8 9 10 11 12 ...

  6. p2150 [NOI2015]寿司晚宴

    传送门 分析 我们发现对于大于$\sqrt(n)$的数每个数最多只会包含一个 所以我们把每个数按照大质数的大小从小到大排序 我们知道对于一种大质数只能被同一个人取 所以f1表示被A取,f2表示被B取 ...

  7. codefirst 最新策略

    http://www.yunjuu.com/info/76058.html 在原有数据库中使用 CodeFirst ,除了第一次添加实体后要立即执行一次 Enable-Migrations add-m ...

  8. HttpUploader6.2-process版本

    1.优化JS逻辑,在上传前先同步相同文件进度,提高多用户上传效率. 2.优化文件块保存逻辑,减少相同文件块的写入操作,减少服务器IO操作,提高上传效率.   js变化: up6.js新增UrlQuer ...

  9. mybatis使用count返回int的方法

    <select id="countByExample" resultType="java.lang.Integer" > select count( ...

  10. poj2299 Ultra-QuickSort(线段树求逆序对)

    Description In this problem, you have to analyze a particular sorting algorithm. The algorithm proce ...