restfulframework引用多对多外键
记录一下工作中遇到的问题
最近在写restfulframework,感觉还是很便利的
首先贴一下文档地址
https://www.django-rest-framework.org/api-guide/filtering/
https://www.django-rest-framework.org/api-guide/serializers/
https://www.django-rest-framework.org/api-guide/relations/#manytomanyfields-with-a-through-model
使用GernricViewSet可以便捷的新增接口,在类中定义queryset指定模型,用serializer_class指定序列化类,用pagination_class指定分页类,再用filter_backends和filter_class做筛选,可以解决大部分curd问题,如下
class GitLabCommit(mixins.ListModelMixin, mixins.RetrieveModelMixin,viewsets.GenericViewSet):
#加RetriveModelMixin可以查询特定模型的信息
queryset = GitCommit.objects.all()
serializer_class = CommitSerializer
pagination_class = MyPageNumberPagination
filter_backends = (DjangoFilterBackend,)
filter_class = UserCommitFilter
其中模型类无需多言,序列化类形如下
class CommitSerializer(serializers.ModelSerializer):
author_email = serializers.CharField(source='author.email')
branch_name = serializers.CharField(source = "branch.branchName")
branch_project_name = serializers.CharField(source="branch.project.projectName") class Meta:
model = GitCommit
fields = "__all__"
fields这里也可以写成元组的形式
fields = ("StaffId","name","staff_name")
分页类类似这样
class MyCursorPagination(pagination.CursorPagination):
"""
Cursor 光标分页 性能高,安全
"""
page_size = 9
ordering = '-update_time'
page_size_query_param = "pages"
max_page_size = 20
cursor_query_description = '页面'
page_size_query_description = '每页条数' class MyPageNumberPagination(pagination.PageNumberPagination):
"""
普通分页,数据量越大性能越差
"""
page_size = 11
page_size_query_param = 'size'
page_query_param = 'page'
max_page_size = 20
这是过滤器类,可以直接定义查找字段或者通过方法进行复杂查找
class UserCommitFilter(filters.FilterSet):
user_id = filters.NumberFilter(field_name='author__StaffId', lookup_expr='exact')
start_date = filters.DateFilter(field_name='commitDate', lookup_expr='gte')
end_date = filters.DateFilter(field_name='commitDate', lookup_expr='lt')
commit_sum = filters.NumberFilter(method="get_sum") def get_sum(self,queryset,name,values):
if values == 1:
return queryset.annotate(total_addLines = Sum("addLines"),total_delLins = Sum("delLines"),total_totalLins = Sum("totalLines"))
这里有一个问题:如果序列化所涉及的模型是关联模型怎么办呢?
可以参考这个
https://zhuanlan.zhihu.com/p/27667372
这里特别说一下,对于多对多模型,可以通过嵌套来进行关联,如下
模型定义
class Staff(BaseTable):
StaffId = models.IntegerField(primary_key=True, help_text="工号")
email = models.CharField(max_length=50,default="",null=True,help_text="邮箱")
name = models.CharField(max_length=50,default="",null=True,help_text="姓名")
department = models.ForeignKey(Department,on_delete=models.CASCADE) def __str__(self):
return "%s:%s"%(self.StaffId,self.name)
class Meta:
db_table = "gitlab_measure_staff" class GitGroup(BaseTable):
id = models.AutoField(primary_key=True, help_text="ID")
name = models.CharField(max_length=100,default="",null=True,help_text="组名称")
members = models.ManyToManyField(Staff) class Meta:
db_table = "gitlab_measure_gitgroup"
序列化
class StaffSerializer(serializers.ModelSerializer):
staff_name = serializers.CharField(source="name")
class Meta:
model = Staff
fields = ("StaffId","name","staff_name") class GitGroupSerializer(serializers.ModelSerializer):
members = StaffSerializer(many=True,read_only=True)
class Meta:
model = GitGroup
fields = ("id","name","members")
如果此时又有一个project类中的group关联到gitgroup,希望在展示的时候展示出组中所有成员该怎么办呢?
这里可以使用depth指定查询的深度
class ProjectSerializer(serializers.ModelSerializer):
gitGroup_name = serializers.CharField(source='gitGroup.name')
gitGroup_id = serializers.CharField(source="gitGroup.id")
department_name = serializers.CharField(source="department.name") class Meta:
model = GitProject
fields = "__all__"
depth = 2
这样在结果中就能看到展示的组和成员了,因为在serilizers.ModelSerializer中的get_field()方法中会根据调用self.build_field,将depth传入,build_field方法会调用self.buid_nested_field方法来,再返回一个ModelSerializer类,再在外层函数中循环调用来获取层层对象,最多不超过10层
for field_name in field_names:
# If the field is explicitly declared on the class then use that.
if field_name in declared_fields:
fields[field_name] = declared_fields[field_name]
continue extra_field_kwargs = extra_kwargs.get(field_name, {})
source = extra_field_kwargs.get('source', '*')
if source == '*':
source = field_name # Determine the serializer field class and keyword arguments.
field_class, field_kwargs = self.build_field(
source, info, model, depth
) # Include any kwargs defined in `Meta.extra_kwargs`
field_kwargs = self.include_extra_kwargs(
field_kwargs, extra_field_kwargs
) # Create the serializer field.
fields[field_name] = field_class(**field_kwargs)
def build_field(self, field_name, info, model_class, nested_depth):
"""
Return a two tuple of (cls, kwargs) to build a serializer field with.
"""
if field_name in info.fields_and_pk:
model_field = info.fields_and_pk[field_name]
return self.build_standard_field(field_name, model_field) elif field_name in info.relations:
relation_info = info.relations[field_name]
if not nested_depth:
return self.build_relational_field(field_name, relation_info)
else:
return self.build_nested_field(field_name, relation_info, nested_depth) elif hasattr(model_class, field_name):
return self.build_property_field(field_name, model_class) elif field_name == self.url_field_name:
return self.build_url_field(field_name, model_class) return self.build_unknown_field(field_name, model_class)
def build_nested_field(self, field_name, relation_info, nested_depth):
"""
Create nested fields for forward and reverse relationships.
"""
class NestedSerializer(ModelSerializer):
class Meta:
model = relation_info.related_model
depth = nested_depth - 1
fields = '__all__' field_class = NestedSerializer
field_kwargs = get_nested_relation_kwargs(relation_info) return field_class, field_kwargs
restfulframework引用多对多外键的更多相关文章
- Python sqlalchemy orm 多对多外键关联
多对多外键关联 注:使用三张表进行对应关联 实现代码: # 创建3个表 配置外键关联 # 调用Column创建字段 加类型 from sqlalchemy import Table, Column, ...
- Django框架表关系外键-多对多外键(增删改查)-正反向的概率-多表查询(子查询与联表查询)
目录 一:表关系外键 1.提前创建表关系 2.目前只剩 书籍表和 书籍作者表没创建信息. 3.增 4.删 5.修改 二:多对多外键增删改查 1.给书籍绑定作者 2.删 3.修改 4.清空 三:正反向的 ...
- 测试脚本配置、ORM必知必会13条、双下划线查询、一对多外键关系、多对多外键关系、多表查询
测试脚本配置 ''' 当你只是想测试django中的某一个文件内容 那么你可以不用书写前后端交互的形式而是直接写一个测试脚本即可 脚本代码无论是写在应用下的test.py还是单独开设py文件都可以 ' ...
- postgreSQL外键引用查询 查询外键被那些表占用
根据一个表名,查询所有外键引用它的表,以及那些外键的列名key_column_usage(系统列信息表),pg_constraint(系统所有约束表) SELECT x.table_name, x.c ...
- Python SQLAlchemy多对多外键关联时表结构
# 创建多对多表结构 from sqlalchemy.ext.declarative import declarative_base from sqlalchemy.engine import cre ...
- Constraint6:更新外键约束(Foreign Key Constraint)的引用列
在SQL Server中,表之间存在引用关系,引用关系通过创建外键约束(Foreign Key Constraint)实现.如果一个Table中的column被其他Table引用,那么该表是参考表,或 ...
- SQLSERVER清空(Truncate)被外键引用的数据表
前言:我们知道SQLSERVER清空数据表有两种方式Delete和Truncate,当然两者的不同大家也都知道(不清楚的可以MSDN).不过这个错误“Cannot truncate table be ...
- 第二百八十节,MySQL数据库-外键链表之一对多,多对多
MySQL数据库-外键链表之一对多,多对多 外键链表之一对多 外键链表:就是a表通过外键连接b表的主键,建立链表关系,需要注意的是a表外键字段类型,必须与要关联的b表的主键字段类型一致,否则无法创建索 ...
- MySQL数据库之-foreign key 外键(一对多、多对多、一对一)、修改表、复制表
摘要: 外键 一对多 外键 多对多 外键 一对一 --------------------------------------------------------------------------- ...
随机推荐
- jmeter beanshell 从文件中获取随机参数
loadruner 参数化有个功能,可以设置在脚本每次出现参数时,自动更换参数值.在做jmeter自动化测试过程中,同一个请求中出现多个参数值,如一个接口可以添加n个信息的请求 [ { "n ...
- (转)Ngx_Lua使用分享
原文:https://www.cnblogs.com/yanzi-meng/p/9450999.html ngx_lua 模块详细讲解(基于openresty)---https://www.cnblo ...
- Java12新特性 -- 增强G1,自动返回未用堆内存给操作系统
Java 12 中增强了 G1 垃圾收集器关于混合收集集合的处理策略,这节主要介绍在 Java 12 中同时也对 G1垃圾回收器进行了改进,使其能够在空闲时自动将 Java 堆内存返还给操作系统,这也 ...
- Centos7环境下部署搭建discuz论坛
1.首先搭建lnmp环境 2.从官网复制git地址(https://gitee.com/ComsenzDiscuz/DiscuzX),在服务器上安装git命令 yum install git -y ...
- Nodejs接收图片base64格式保存为文件
base64的形式为“....”:当接收到上边的内容后,需要将data ...
- Dubbo_异常_服务注册运行正常但是Dubbo-Admin看不到服务
出自:https://www.cnblogs.com/gossip/p/6021698.html 一.背景: 1.Dubbo服务正常注册到ZooKeeper 2.客户端调用Dubbo服务正常 二.原因 ...
- 最常见的Java面试题及答案汇总(三)
上一篇:最常见的Java面试题及答案汇总(二) 多线程 35. 并行和并发有什么区别? 并行是指两个或者多个事件在同一时刻发生:而并发是指两个或多个事件在同一时间间隔发生. 并行是在不同实体上的多个事 ...
- pidstat 命令详解
pidstat 概述 pidstat是sysstat工具的一个命令,用于监控全部或指定进程的cpu.内存.线程.设备IO等系统资源的占用情况.pidstat首次运行时显示自系统启动开始的各项统计信息, ...
- python的mysql数据库操作
python操作mysql数据库 Python 标准数据库接口为 Python DB-API,Python DB-API为开发人员提供了数据库应用编程接口. Python 数据库接口支持非常多的数据库 ...
- redis源码分析(四)--aof持久化
Redis aof持久化 Redis支持两种持久化方式:rdb与aof,上一篇文章中已经大致介绍了rdb的持久化实现,这篇文章主要介绍aof实现. 与rdb方式相比,aof会使用更多的存储空间,因为它 ...