Django restframe 视图函数以及ModelSerializer的使用
建立model数据库
from django.db import models
__all__ = ['Book', 'Publisher', 'Author']
# Create your models here.
class Book(models.Model):
title = models.CharField(max_length=32)
CHOICES = ((1, 'python'), (2, 'linux'), (3, 'go'))
category = models.IntegerField(choices=CHOICES, default=1, verbose_name='分类')
pub_time = models.DateField(verbose_name='出版日期')
publisher = models.ForeignKey(to='Publisher', verbose_name='出版社')
authors = models.ManyToManyField(to='Author', verbose_name='作者')
class Meta:
db_table = 'book'
verbose_name = '书籍表'
verbose_name_plural = verbose_name
def __str__(self):
return self.title
class Publisher(models.Model):
name = models.CharField(max_length=32, verbose_name='出版社')
address = models.CharField(max_length=32, verbose_name='地址')
class Meta:
db_table = 'publisher'
verbose_name = '出版社表表'
verbose_name_plural = verbose_name
def __str__(self):
return self.name
class Author(models.Model):
name = models.CharField(max_length=32, verbose_name='作者')
country = models.CharField(max_length=32, verbose_name='国家')
class Meta:
db_table = 'author'
verbose_name = '作者表'
verbose_name_plural = verbose_name
def __str__(self):
return self.name
数据库部分一共三张表,书籍表以及作则和出版社表
admin注册
from django.contrib import admin
from SerDemo import models
# Register your models here.
for table in models.__all__:
admin.site.register(getattr(models,table))
这里用到了小技巧,利用导入的models.__all__循环批量注册model模型
序列器的实现
在项目下新建serializers,py文件
from rest_framework import serializers
from SerDemo.models import Book, Author, Publisher
# 自定义检验函数在字段参数validators中作为参数添加
def my_validate(value):
if "敏感词汇" in value.lower():
raise serializers.ValidationError("输入的信息含有敏感词汇")
return value
### 创建django序列化器类,继承serializers.ModelSerializer字段可自动关联
class BookSerializer(serializers.ModelSerializer):
## 需要重写的字段,SerializerMethodField 会去找get_字段名的函数并执行获取返回值作为字段值
category_read = serializers.SerializerMethodField(read_only=True,validators=[my_validate,])
publisher_read = serializers.SerializerMethodField(read_only=True)
authors_read = serializers.SerializerMethodField(read_only=True)
# category_read的钩子函数,返回要显示的值,注意obj是每个要显示到前端的数据对象
def get_category_read(self,obj):
return obj.get_category_display()
# publisher_read的钩子函数,返回要显示的值
def get_publisher_read(self,obj):
return {'id':obj.publisher_id,'title':obj.publisher.name}
def get_authors_read(self,obj):
ret = [{'id':i.id,'name':i.name} for i in obj.authors.all()]
return ret
## 全局校验钩子函数
def validate(self, attrs):
# attrs 包含所有字段的数据
if 'se' in attrs['title']:
#如果敏感词汇在title字段中则抛出错误
raise serializers.ValidationError('含有敏感词汇')
# 否则返回原信息
return attrs
# 局部校验钩子函数validate_+字段名
def validate_title(self, data):
if 'se' in data:
raise serializers.ValidationError('含有敏感词汇')
return data
# 序列化器的元信息
class Meta:
# 绑定的数据表model
model=Book
#要展示的表的字段
fields = '__all__'
# depth = 1
# depth 让你所有的外键关系变成read_only=True,不建议使用
# 额外要添加的字段的属性参数
extra_kwargs = {
'category':{'write_only':True},
'publisher': {'write_only': True},
'authors':{'write_only':True},
}
路由以及视图函数部分
from django.conf.urls import url
from SerDemo.views import BooklistView,EditBookView
urlpatterns = [
# 展示书籍信息路由
url(r'^booklist/',BooklistView.as_view()),
#编辑书籍路由,接受id参数
url(r'^editbook/(?P<id>\d+)/',EditBookView.as_view())
]
视图部分
展示所有书籍数据和添加数据视图,
class BooklistView(ListCreateModelMixin):
#get 请求发送所有数据给前端
def get(self, request):
# 获取数据库中所有数据对象
query_set = Book.objects.all()
# 将queryset对象传给序列化器实例化,many=True告诉序列化器这是多个对象
ser_obj = BookSerializer(query_set, many=True)
# 所有要显示的数据都在ser_obj.data中用restframe的Response返回
return Response(ser_obj.data)
# post请求为添加数据
def post(self, request):
# 序列化前端传过来的数据(request.data)
ser_obj = BookSerializer(data=request.data)
# 校验各字段数据是否符合要求
if ser_obj.is_valid():
ser_obj.save()
return Response(ser_obj.data)
else:
return Response(ser_obj.errors)
编辑数据和删除书籍部分
class EditBookView(UpdateDestroyModelMixin):
# 展示当前要编辑的数据对象
def get(self, request, id):
#根据url中的id参数获取要编辑数据对象
book_obj = Book.objects.filter(id=id).first()
if book_obj:
# 返回给前端页面
ser_obj = BookSerializer(book_obj)
return Response(ser_obj.data)
return Response('没有')
#put请求对应修改数据
def put(self, request, id):
#根据url中的id参数获取要编辑数据对象
book_obj = Book.objects.filter(id=id).first()
if book_obj:
#将前端提交的数据和要修改的对象传给序列化器partial=True允许只修改部分字段数据
ser_obj = BookSerializer(data=request.data, instance=book_obj, partial=True)
if ser_obj.is_valid():
ser_obj.save()
return Response(ser_obj.data)
else:
return Response(ser_obj.errors)
def delete(self,request,id):
book_obj = Book.objects.filter(id=id).first()
if book_obj:
book_obj.delete()
return Response('')
return Response('没有次对象')
模仿djngo restframe源码封装各个方法
from django.shortcuts import render, HttpResponse, redirect
from django.http import JsonResponse
import json
from SerDemo.models import Author, Publisher, Book
from django import views
from rest_framework import serializers
from rest_framework.views import APIView
from rest_framework.response import Response
from SerDemo.serializers import BookSerializer
# 把每个方法抽离出来
class GenericAPIView(APIView):
query_set = None
serializer_class = None
def _get_queryset(self):
return self.query_set
def _get_serializer(self,*args,**kwargs):
return self.serializer_class(*args,**kwargs)
class ListModelMixin:
def list(self):
query_set = self._get_queryset().all()
ser_obj = self._get_serializer(query_set, many=True)
return Response(ser_obj.data)
class CreateModelMixin:
def create(self,request):
ser_obj = self._get_serializer(data=request.data)
if ser_obj.is_valid():
ser_obj.save()
return Response(ser_obj.data)
else:
return Response(ser_obj.errors)
class RetrieveModelMixin:
def retrieve(self,id):
book_obj = self._get_queryset().filter(id=id).first()
if book_obj:
ser_obj = self._get_serializer(book_obj)
return Response(ser_obj.data)
return Response('没有')
class UpdateModelMixin:
def update(self,request,id):
book_obj = self._get_queryset().filter(id=id).first()
if book_obj:
ser_obj = self._get_serializer(data=request.data, instance=book_obj, partial=True)
if ser_obj.is_valid():
ser_obj.save()
return Response(ser_obj.data)
else:
return Response(ser_obj.errors)
class DestroyModelMixin:
def destroy(self,id):
book_obj = self._get_queryset().filter(id=id).first()
if book_obj:
book_obj.delete()
return Response('')
return Response('没有次对象')
class ListCreateModelMixin(GenericAPIView,ListModelMixin, CreateModelMixin):
pass
class UpdateDestroyModelMixin(GenericAPIView,DestroyModelMixin,UpdateModelMixin,RetrieveModelMixin):
pass
class BooklistView(ListCreateModelMixin):
query_set = Book.objects.all()
serializer_class = BookSerializer
def get(self, request):
# query_set = Book.objects.all()
# ser_obj = BookSerializer(query_set, many=True)
# return Response(ser_obj.data)
return self.list()
def post(self, request):
# ser_obj = BookSerializer(data=request.data)
# if ser_obj.is_valid():
# ser_obj.save()
# return Response(ser_obj.data)
# else:
# return Response(ser_obj.errors)
return self.create(request)
class EditBookView(UpdateDestroyModelMixin):
query_set = Book.objects.all()
serializer_class = BookSerializer
def get(self, request, id):
# book_obj = Book.objects.filter(id=id).first()
# if book_obj:
# ser_obj = BookSerializer(book_obj)
# return Response(ser_obj.data)
# return Response('没有')
return self.retrieve(id)
def put(self, request, id):
# book_obj = Book.objects.filter(id=id).first()
# if book_obj:
# ser_obj = BookSerializer(data=request.data, instance=book_obj, partial=True)
# if ser_obj.is_valid():
# ser_obj.save()
# return Response(ser_obj.data)
# else:
# return Response(ser_obj.errors)
return self.update(request,id)
def delete(self,request,id):
# book_obj = Book.objects.filter(id=id).first()
# if book_obj:
# book_obj.delete()
# return Response('')
# return Response('没有次对象')
return self.destroy(id)
Django restframe 视图函数以及ModelSerializer的使用的更多相关文章
- Django之视图函数总结
Django之视图函数总结 HttpRequest与HttpResponse http请求中产生两个核心对象: HttpRequest对象:用户请求相关的所有信息(对象) HttpResponse对象 ...
- django views视图函数返回值 return redirect httpresponse总结
django views视图函数返回值 return redirect render httpresponse总结
- Django 定义视图函数
Django 定义视图函数 一.接收内容及文件处理 1.接收分类 # 获取数据 request.GET # 提交数据 request.POST # 获取文件 request.FILES 2.check ...
- 03 Django之视图函数
一.Django的视图函数view 一个视图函数(类),简称视图,是一个简单的Python函数(类),它接受WEB请求并返回Web响应. 响应可以是一张网页的HTML内容,一个重定向,一个404错误, ...
- django的视图函数
一.视图函数view 视图函数是接收一个请求(request对象),并返回响应的函数 1. HttpResponse响应请求 这个方法是返回字符串一类的,可以识别标签 2. render响应请求 re ...
- Django的视图函数和路由系统中一些没有用过的小点
1.request对象 print("返回用户访问的url,但是不包括域名",request.path_info) print("返回请求的方法,全大写",re ...
- django views视图函数
Django views.py视图文件 一. 创建views.py文件,在工程文件夹根目录创建views.py视图文件,其实任意文件名都可以,使用views是为了遵循传统. 注:所有的views函数都 ...
- Django中非视图函数获取用户对象
今天遇到了一个问题:在Django中怎么从非视图函数中获取用户对象?怎么保证不同的请求获取到不同的用户对象? 平常我们获取用户对象使用的是: request.user 不得不说,这确实很方便. 但是, ...
- django中视图函数的FBV和CBV
1.什么是FBV和CBV FBV是指视图函数以普通函数的形式:CBV是指视图函数以类的方式. 2.普通FBV形式 def index(request): return HttpResponse('in ...
随机推荐
- Windows Community Toolkit 3.0 - UniformGrid
概述 UniformGrid 控件是一个响应式的布局控件,允许把 items 排列在一组均匀分布的行或列中,以填充整体的可用显示空间,形成均匀的多个网格.默认情况下,网格中的每个单元格大小相同. 这是 ...
- 苹果 icloud 把我 ipad min 所有照片丢失
苹果 icloud 把我 ipad min 所有照片丢失,大概发生在 '云上贵州' 之后! 发帖纪念--- 求个说法---
- Node+GitLab实现小程序CI系统
为什么要实现自动部署 小程序开发迭代里,有以下几个个头痛的问题, 如何准确并快速的的把小程序上传去后台,并让测试人员进行测试? 测试同事找开发要二维码,效率较低 本地生成的二维码会出现携带本地代码.未 ...
- vue内置组件 transition 和 keep-alive 使用
1.transition name - string,用于自动生成 CSS 过渡类名.例如:name: 'fade' 将自动拓展为.fade-enter,.fade-enter-active等.默认类 ...
- Vue(二)基础
01-vue的起步 1.引包 a) 直接下载,并用<script>标签引入 b) CDN方式引入: <script src="https://cdn.bootcss.com ...
- 如何用 Node.js 和 Elasticsearch 构建搜索引擎
Elasticsearch 是一款开源的搜索引擎,由于其高性能和分布式系统架构而备受关注.本文将讨论其关键特性,并手把手教你如何用它创建 Node.js 搜索引擎. Elasticsearch 概述 ...
- 2017软工实践K班总结
回首一学期的软工实践,从暑假开始陆续布置作业,经历个人.结对与团队等大小作业.也经历了不少同学被吓跑.第一周就退选的情况,能坚持下来的都是胜利者,至少你们有一颗愿意挑战的心.首先感谢助教谢涛付出的巨大 ...
- python中变量、函数、类名、模块名等命名方式
摘要:模块名:小写字母,单词之间用_分割ad_stats.py包名:和模块名一样类名:单词首字母大写AdStatsConfigUtil全局变量名(类变量,在java中相当于static变量):大写字母 ...
- Python迭代器与格式化
三元运算 对if-else判断的简写 >>> age = 18 >>> res = "You are so young!" if age < ...
- Day2 Numerical simulation of optical wave propagation之标量衍射理论基本原理(二)
2.麦克斯韦方程组的简单行波解 讨论通过线性.各向同性.均匀.无色散.无限电荷和电流的电介质材料的光波传输.在这种情况下,介质具有如下属性: (1)推导获得波动方程( 由麦克斯韦方程组导出的.描述电磁 ...