django框架之drf:04、序列化器常用字段及参数,序列化器高级用法之source、定制字段数据的两种方法、多表关联反序列化的保存、ModelSerializer的使用
Django框架之drf
一、序列化器常用字段及参数
# 序列化类---》字段类 CharField,除此之外还有哪些其他的
# 序列化类---》字段类,字段类上,传属性的 ,序列化类上,也可以写属性
models.CharField(max_length=32)
1、常用字段
字段 | 字段构造方式 |
---|---|
BooleanField | BooleanField() |
NullBooleanField | CharField(max_length=None, min_length=None, allow_blank=False, trim_whitespace=True) |
CharField | CharField(max_length=None, min_length=None, allow_blank=False, trim_whitespace=True) |
EmailField | EmailField(max_length=None, min_length=None, allow_blank=False) |
RegexField | RegexField(regex, max_length=None, min_length=None, allow_blank=False) |
SlugField | SlugField(maxlength=50, min_length=None, allow_blank=False) 正则字段,验证正则模式 [a-zA-Z0-9-]+ |
URLField | URLField(max_length=200, min_length=None, allow_blank=False) |
UUIDField | UUIDField(format=’hex_verbose’) format: 1) ‘hex_verbose’ 如"5ce0e9a5-5ffa-654b-cee0-1238041fb31a" 2) ‘hex’ 如 “5ce0e9a55ffa654bcee01238041fb31a” 3)‘int’ - 如: “123456789012312313134124512351145145114” 4)‘urn’ 如: “urn:uuid:5ce0e9a5-5ffa-654b-cee0-1238041fb31a” |
IPAddressField | IPAddressField(protocol=’both’, unpack_ipv4=False, **options) |
IntegerField | IntegerField(max_value=None, min_value=None) |
FloatField | FloatField(max_value=None, min_value=None) |
DecimalField | DecimalField(max_digits, decimal_places, coerce_to_string=None, max_value=None, min_value=None) max_digits: 最多位数 decimal_palces: 小数点位置 |
DateTimeField | DateTimeField(format=api_settings.DATETIME_FORMAT, input_formats=None) |
DateField | DateField(format=api_settings.DATE_FORMAT, input_formats=None) |
TimeField | TimeField(format=api_settings.TIME_FORMAT, input_formats=None) |
DurationField | DurationField() |
ChoiceField | ChoiceField(choices) choices与Django的用法相同 |
MultipleChoiceField | MultipleChoiceField(choices) |
FileField(max_length=None, allow_empty_file=False, use_url=UPLOADED_FILES_USE_URL) | |
ImageField | ImageField(max_length=None, allow_empty_file=False, use_url=UPLOADED_FILES_USE_URL) |
ListField | ListField(child=, min_length=None, max_length=None) |
DictField | DictField(child=) |
2、常用字段参数
选项参数
参数名 | 说明 |
---|---|
max_length | 最大长度 |
min_lenght | 最小长度 |
allow_blank | 是否允许为空 |
trim_whitespace | 是否截断空白字符 |
max_value | 最大值 |
min_value | 最小值 |
通用参数
参数名称 | 说明 |
---|---|
read_only | 表明该字段仅用于序列化输出,默认False |
write_only | 表明该字段仅用于反序列化输入,默认False |
required | 表明该字段在反序列化时必须输入,默认True |
default | 反序列化时使用的默认值 |
allow_null | 表明该字段是否允许传入None,默认False |
validators | 该字段使用的验证器 |
error_messages | 包含错误编号与错误信息的字典 |
label | 用于HTML展示API页面时,显示的字段名称 |
help_text | 用于HTML展示API页面时,显示的字段帮助提示信息 |
3、字段参数针对性分类
选项参数:
# CharField及其子类的(EmailField) ---》反序列化的校验,字段自己的规则
max_length 最大长度
min_lenght 最小长度
allow_blank 是否允许为空
trim_whitespace 是否截断空白字符
# IntegerField
max_value 最小值
min_value 最大值
# 所有字段类都有的
required 表明该字段在反序列化时必须输入,默认True
default 反序列化时使用的默认值
allow_null 表明该字段是否允许传入None,默认False
validators 该字段使用的验证器
----看一眼忘掉-----
error_messages 包含错误编号与错误信息的字典
label 用于HTML展示API页面时,显示的字段名称
help_text 用于HTML展示API页面时,显示的字段帮助提示信息
# 重点:
read_only 表明该字段仅用于序列化输出,默认False
write_only 表明该字段仅用于反序列化输入,默认False
## 反序列化校验执行流程
-1 先执行字段自己的校验规则----》最大长度,最小长度,是否为空,是否必填,最小数字。。。。
-2 validators=[方法,] ----》单独给这个字段加校验规则
name=serializers.CharField(validators=[方法,])
-3 局部钩子校验规则
-4 全局钩子校验规则
二、序列化器高级用法之source
准备工作
### 创建关联表
class Book(models.Model):
name = models.CharField(max_length=32)
price = models.CharField(max_length=32)
publish = models.ForeignKey(to='Publish', on_delete=models.CASCADE)
authors = models.ManyToManyField(to='Author')
class Publish(models.Model):
name = models.CharField(max_length=32)
addr = models.CharField(max_length=32)
class Author(models.Model):
name = models.CharField(max_length=32)
phone = models.CharField(max_length=11)
# 迁移,录入数据
1、定制字段名
在我们编写序列化器的时候,序列化类中的字段名字对应的是模型层下表名内对应的字段名,但有的时候我们需要确保数据的安全,并不想直接将真实的字段名返回给前端用户查看,这个时候我们就可以利用source参数来将返回给前端的字段名进行修改
source参数:可以指定序列化字段的名字
class BookSerializer(serializers.Serializer):
# 自有字段: 直接写字段名
book_name = serializers.CharField(source='name')
book_price = serializers.CharField(source='price')
# 外键字段:多对多,字段名字修改了,但是数据内容没办法显示
book_author = serializers.CharField(source='author.all')
# 外键字段:一对多,可以显示
book_publish = serializers.CharField(source='publish.name')
三、定制字段数据的两种的方法
定制关联字段的显示形式
- 一对多:显示字典
- 多对多:显示列表内套字典
1、在序列化器类中定制
使用:SerializerMethodField字段定制
from rest_framework import serializers
class BookSerializer(serializers.Serializer):
# 自有字段
book_name = serializers.CharField(source='name')
book_price = serializers.CharField(source='price')
# 外键字段
book_publish = serializers.SerializerMethodField()
def get_book_publish(self, obj):
return {'name': obj.publish.name, 'address': obj.publish.address}
book_author = serializers.SerializerMethodField()
def get_book_author(self, obj):
book_data_list = [{'name': author_obj.name, 'phone': author_obj.phone} for author_obj in obj.author.all()]
return book_data_list
2、在模型表中定制
在models.py文件下表的类中定制
### models.py 表的类中编写方法
from django.db import models
class Book(models.Model):
name = models.CharField(max_length=32)
price = models.CharField(max_length=32)
publish = models.ForeignKey(to='Publish', on_delete=models.CASCADE)
def publish_detail(self):
return {'name': self.publish.name, 'phone': self.publish.address}
author = models.ManyToManyField(to='Author')
def author_list(self):
book_data_list = [{'name': author_obj.name, 'phone': author_obj.phone} for author_obj in self.author.all()]
return book_data_list
### Serialiaer.py 序列化器中编写方法
from rest_framework import serializers
class BookSerializer(serializers.Serializer):
# 自有字段
book_name = serializers.CharField(source='name')
book_price = serializers.CharField(source='price')
# 外键字段
publish_detail = serializers.DictField()
author_list = serializers.ListField()
四、多表关联反序列化保存
前端传入数据格式
# 前端传入的数据格式:
{'name':'红楼梦','price':19,'publish':1,'authors':[1,2]}
1、新增接口
### view.py 视图类
class BookView(APIView):
# 新增
def post(self, request):
ser_obj = BookSerializer(data=request.data)
if ser_obj.is_valid():
ser_obj.save()
return Response({'code': 100, 'msg': '新增图书成功', 'result': ser_obj.data})
return Response({'code': 101, 'msg': ser_obj.errors})
### serializer.py 序列化器类
class BookSerializer(serializers.Serializer):
# 自有字段
name = serializers.CharField()
price = serializers.CharField()
# 设置write_only参数,只作为反序列化使用
publish = serializers.CharField(write_only=True)
author = serializers.ListField(write_only=True)
# 外键字段,设置read_only参数,只作为序列化使用
publish_detail = serializers.DictField(read_only=True)
author_list = serializers.ListField(read_only=True)
# 新增
def create(self, validated_data):
# 使用反序列化后的数据创建新的图书
new_book_obj = Book.objects.create(name=validated_data.get('name'), price=validated_data.get('price') , publish_id=validated_data.get('publish'))
# 作者外键字段同步更新
new_book_obj.author.add(*validated_data.get('author'))
return new_book_obj
2、修改接口
### view.py 视图类:
class BookDetailView(APIView):
# 修改
def put(self, request, pk):
# 获取指定图书
target_book_obj = Book.objects.filter(pk=pk).first()
if target_book_obj:
ser_obj = BookSerializer(data=request.data, instance=target_book_obj)
if ser_obj.is_valid():
ser_obj.save()
return Response(ser_obj.data)
return Response({'code': 101, 'msg': ser_obj.errors})
return Response({'code': 101, 'msg': '图书不存在'})
### serializer.py 序列化器类:
class BookSerializer(serializers.Serializer):
# 自有字段
name = serializers.CharField()
price = serializers.CharField()
# 设置write_only参数,只作为反序列化使用
publish = serializers.CharField(write_only=True)
author = serializers.ListField(write_only=True)
# 外键字段,设置read_only参数,只作为序列化使用
publish_detail = serializers.DictField(read_only=True)
author_list = serializers.ListField(read_only=True)
# 修改
def update(self, instance, validated_data):
instance.name = validated_data.get('name')
instance.price = validated_data.get('price')
instance.publish_id = validated_data.get('publish')
instance.author.clear()
instance.author.add(*validated_data.get('author'))
instance.save()
return instance
五、反序列化字段校验(总结)
反序列化字段校验(共四层)
1、自有字段:可直接在字段后方参数填写校验规则
2、validators参数:同样在字段后方参数内填写,通过绑定函数体代码进行校验
3、局部钩子
4、全局钩子
六、ModelSerializer的使用
ModelSerializer继承自Serializer,帮助我们完成了很多操作
特点:
- 和表模型强关联
- 帮助我们完成很多请求,不用再create和update
使用方法:
class BookSerializer(serializers.ModelSerializer):
# 控制字段的校验
class Meta:
# 与表进行关联
model = Book
# 填写__all__默认序列全部字段,如果Meta写了__all__ ,就相当于,复制了表模型中的所有字段,放在了这里,做了个映射
# fields = '__all__'
# 填写列表是校验部分字段
fields = ['name', 'price', 'publish', 'author', 'publish_detail', 'author_list']
# 给字段添加校验或限制
extra_kwargs = {
'name': {'max_length': 3},
'price': {'min_length': 2},
'publish': {'write_only': True},
'author': {'write_only': True},
'author_list': {'read_only': True},
'publish_detail': {'read_only': True},
}
# 假如Meta类中已经对字段进行校验,任然可以在外部(内部的校验失效)重写校验,优先级高于Meta内部的校验
name = serializers.CharField(max_length=8)
# 同理,针对外键字段的方法也可以在外部重写
# book_publish = serializers.SerializerMethodField()
#
# def get_book_publish(self, obj):
# return {'name': obj.publish.name, 'address': obj.publish.address}
#
# book_author = serializers.SerializerMethodField()
#
# def get_book_author(self, obj):
# book_data_list = [{'name': author_obj.name, 'phone': author_obj.phone} for author_obj in obj.author.all()]
# return book_data_list
# 钩子函数(不会影响,正常编写即可)
def validate_name(self, name):
if name.startswith('sb'):
raise ValidationError('不能sb')
else:
return name
django框架之drf:04、序列化器常用字段及参数,序列化器高级用法之source、定制字段数据的两种方法、多表关联反序列化的保存、ModelSerializer的使用的更多相关文章
- laravel5.5框架中视图间如何共享数据?视图间共享数据的两种方法
laravel框架中视图间共享数据有两种,一种是用视图门面share()方法实现,另一种是用视图门面composer() 方法实现,那么,两种方法的实现究竟是怎样的呢?让我们来看一看接下来的文章内容. ...
- java 获取键盘输入常用的两种方法
java 获取键盘输入常用的两种方法 方法1: 通过 Scanner Scanner input = new Scanner(System.in); String s = input.nextLine ...
- Django框架之DRF 基于mixins来封装的视图
基础视图 示例环境搭建:新建一个Django项目,连接Mysql数据库,配置路由.视图函数.序列化单独创建py文件 # 配置路由 from django.conf.urls import url fr ...
- Django框架(十二)—— 补充:inclusion_tag、defer、only、choice、事务、创建多对多的第三张表
目录 补充:inclusion_tag.defer.only.choice.事务.创建多对多的第三张表 一.inclusion_tag 1.作用 2.使用 二.defer与only 1.定义 2.使用 ...
- yii2.0高级框架配置时打开init.bat秒退的解决方法 (两种方法)
第一种: 这几天刚接触到yii2.0框架,在配置advanced版本时运行init.bat初始化文件时老是闪退: 用cmd运行该文件时显示:The OpenSSL PHP extension is r ...
- django基础之day05,F与Q查询,Q查询的高级用法
#F与Q查询 #*************************** F 查询 ******************** # F 查询数据库中的其他字段!!! #1.查询库存数大于卖出数的书籍 fr ...
- Django 构建模板form表单的两种方法
通常情况下,我们想构建一张表单时会在模板文件login.html中写入 <form action="/your-name/" method="post"& ...
- Django框架之DRF get post put delete 使用简单示例 (利用序列化反序列化)
路由配置 # 路由 from django.conf.urls import url from django.contrib import admin from app01 import views ...
- Django框架之DRF APIView Serializer
一.APIView 我们在使用DjangoRestfulFramework的时候会将每个视图类继承APIView,取代原生Django的View类 APIView的流程分析: rest_framewo ...
- Django框架之DRF 认证组件源码分析、权限组件源码分析、频率组件源码分析
认证组件 权限组件 频率组件
随机推荐
- Vue2基础知识学习
Vue2基础知识学习 01.初识 new Vue({ el: '#root', //用于指定当前Vue实例为哪个容器服务,值通常为css选择器符 data () { return { } } }); ...
- CSP-S2022 游寄
前言:最后确实寄了,因为疫情,都没考成. \(8.26\) 占坑. \(8.23\) 参加浴谷月赛初赛模拟,报的 \(S\) 组,只有 \(71\) 分. \(8.25\) \(AK\) 了同学出的比 ...
- 【iOS逆向与安全】frida-trace入门
前言 frida-trace是一个用于动态跟踪函数调用的工具.支持android和ios.安装教程请参考官网.工欲善其事必先利其器.本文将以某App为示范,演示frida-trace的各种方法在iOS ...
- Linux系统安装 tftp服务 NFS服务
安装tftp服务 安装 sudo apt-get install tftp-hpa tftpd-hpa 配置文件 # /etc/default/tftpd-hpa TFTP_USERNAME=&quo ...
- Web Api出现500 Internal Server Error 错误
在测试环境一切正常,但是部署到了生产环境发现一直报错.查询网上的方法设置了权限等等.都没有解决 原来发现是数据库连接字符串的问题.只需要把数据库连接字符串修改正确即可!
- Linux配置ipv6脚本
#!/bin/bash REMOTE_IP6="2001:da8:900c:eeee:0:5efe" REMOTE_IP4="" #填你自己学校的路由隧道的ip ...
- 关于linux上实现arp攻击截取密码
前言 这几天简单的研究了一下arp攻击,有一些进展,记录一下 环境准备 这里我是利用arpspoof 这个软件简单实现arp攻击,这个命令是属于dsniff 软件包中的 所以首先安装软件 sudo a ...
- C++编程笔记(多线程学习)
目录 一.线程创建 二.线程的相关操作 2.1 join 2.2 detach 2.3 joinable 三.线程参数 3.1传参所引发的资源回收问题 3.2 将对象的成员函数作为入口函数 四.线程的 ...
- CompletionService 使用小结
本文为博主原创,转载请注明出处: 实现异步任务时,经常使用 FutureTask 来实现:一个简单的示例代码如下: public static void main(String[] args) thr ...
- 【每日一题】【使用list&使用辅助栈实现】2022年2月11日-NC90 包含min函数的栈
描述定义栈的数据结构,请在该类型中实现一个能够得到栈中所含最小元素的 min 函数,输入操作时保证 pop.top 和 min 函数操作时,栈中一定有元素. 此栈包含的方法有:push(value): ...