【Django学习】(十一)APIView_请求与响应_GenericAPIView

  •  继承DRF中APIView之后,那么当前视图就具备了认证、授权、限流等功能
  •  继承DRF中APIView之后,每一个实例方法中的request为Request对象
  •  Request类拓展了Django中的HttpRequest类,具备很多额外优秀的功能
  • Request类与HttpRequest类中的所有功能兼容
  • 查询字符串参数:request.GET、request.query_param推荐
  • 获取json格式参数:request.body(还需要decode解码)、request.data(直接获取到字典)
  • 获取www-form参数:request.POST、request.data
  • 获取multipart/form-data:request.body、request.POST、request.data

一、APIView

1、Request_解析类

from rest_framework.views import APIView

 视图集中引用APIView

class ProjectsDetailViews(APIView):
.
.
.class ProjectsViews(APIView):# 创建数据def post(self, request):ret = {"msg": "传参异常","code": 404}# json_str = request.body.decode('utf-8')# try:#     data_dict = json.loads(json_str)# 如果入参不是json格式数据,抛出异常# except json.JSONDecodeError:#     return JsonResponse(ret, json_dumps_params={"ensure_ascii": False}, status=404)# 反序列化输入# serializer_obj = ProjectModelSerializer(data=data_dict)serializer_obj = ProjectModelSerializer(data=request.data)if not serializer_obj.is_valid(raise_exception=True):ret.update(serializer_obj.errors)return JsonResponse(ret, json_dumps_params={"ensure_ascii": False}, status=404)serializer_obj.save()return JsonResponse(serializer_obj.data, json_dumps_params={"ensure_ascii": False}, status=200)
  •  上面的代码中注释掉入参为json格式数据的处理,使用drf框架中自带的解析类
    • JSONParser:解析JSON请求内容。
    • FormParser:解析 HTML 表单内容
    • MultiPartParser:解析多部分HTML表单内容,支持文件上传

 设置解析器:

可以使用DEFAULT_PARSER_CLASSES设置全局默认的解析器集。例如,以下设置将仅允许具有JSON内容的请求,而不是JSON或表单数据的默认值。

  • 定义解析器类,用于解析不通的前端参数类型
  • 会自动根据请求头中Content-Type来解析参数
  • 无论前端传递这三种参数中的哪一种参数,都可以使用request.data去获取

 本项目中的配置文件添加配置:

 

 发起post请求:

 

 如果发起的数据请求格式在解析类中没有定义,则发起请求时返回提示请求格式不支持

 

 

    # 创建数据def post(self, request):ret = {"msg": "传参异常","code": 404}# json_str = request.body.decode('utf-8')# try:#     data_dict = json.loads(json_str)# 如果入参不是json格式数据,抛出异常# except json.JSONDecodeError:#     return JsonResponse(ret, json_dumps_params={"ensure_ascii": False}, status=404)# 反序列化输入# serializer_obj = ProjectModelSerializer(data=data_dict)serializer_obj = ProjectModelSerializer(data=request.data)if not serializer_obj.is_valid(raise_exception=True):ret.update(serializer_obj.errors)return JsonResponse(ret, json_dumps_params={"ensure_ascii": False}, status=404)serializer_obj.save()return JsonResponse(serializer_obj.data, json_dumps_params={"ensure_ascii": False}, status=200)

 指定了解析类后,我们在视图集里就可以不用再手动处理json格式或者其他格式的入参(代码被注释掉的部分) ,drf内部会自动帮我们处理

 

2、Response_渲染类

  • 定义渲染类,用于返回不同类型的数据
    • 会自动根据请求头中Accept进行渲染
    • 如果前端不指定Accept,那么默认返回json格式的数据
    • 如果指定Accept为application/json,那么也会以json数据返回
    • 如果指定Accept为text/html(浏览器发起GET请求会自动指定),那么会以html形式返回
  • 在DRF的视图中,一定要以Response返回 
  • Response类为HttpResponse的子类,具备HttpResponse中的所有功能
  • 可以自动根据请求头中的Accept参数,来返回相应数据格式到前端
  • 第一个参数为Python中的常用数据类型(字典、嵌套字典的列表),尽量使用序列化器类.data
  • status指定响应状态码,content_type指定响应体数据类型
  •  

settings.py配置文件 

 

 指定了渲染类后,我们在views.py视图集里也可以不用再手动处理json格式的出参了,drf会内部自动帮我们处理并返回,所以可以不使用JsonResponse进行返回了。

# return JsonResponse(serializer_obj.data, json_dumps_params={"ensure_ascii": False}, status=200)

 可以使用Response进行返回:

return Response(serializer_obj.data,status=status.HTTP_201_CREATED)

 

二、GenericAPIView

  • GenericAPIView是APIView的子类,具备APIView的所有功能
  • 往往需要指定queryset和serializer_class两个类属性
  • queryset指定当前类视图需要使用的查询集
  • serializer_class指定当前类视图需要使用的序列化器类 

 

import ast
import jsonfrom django.http import HttpResponse, JsonResponse, Http404
from rest_framework.response import Response
from rest_framework import status
from django.views import View
from rest_framework.views import APIView
from rest_framework.generics import GenericAPIView
from .models import ProjectsModel
from django.db import connection
from .serializers import ProjectSerializer, ProjectModelSerializer, ProjectNameSerializer# 能够动态的查询生成的sql语句
connection.queries# Create your views here.# class ProjectsDetailViews(APIView):
class ProjectsDetailViews(GenericAPIView):queryset = ProjectsModel.objects.all()serializer_class = ProjectModelSerializerdef get(self, request, pk):pro = self.get_object()serializer_obj = self.serializer_class(instance=pro)return Response(serializer_obj.data, status=status.HTTP_200_OK)# 更新数据def put(self, request, pk):# 查出对应id的数据query_data = self.get_object()serializer_obj = self.serializer_class(instance=query_data, data=request.data)serializer_obj.is_valid()# 保存更新的数据serializer_obj.save()return Response(serializer_obj.data, status=status.HTTP_201_CREATED)# 删除数据def delete(self, request, pk):ret = {"msg": "删除成功!"}# 根据id查出对应数据query_data = self.get_object()# 删除指定数据query_data.delete()# 一般删除数据的输出为Nonereturn Response(ret,status=status.HTTP_204_NO_CONTENT)# class ProjectsViews(APIView):
class ProjectsViews(GenericAPIView):queryset = ProjectsModel.objects.all()serializer_class = ProjectNameSerializer# 查询全部数据def get(self, request):# 往往不要直接使用queryset类属性,去获取查询集对象# 一般需要使用self.get_queryset(),去获取查询集对象,因为更加灵活,后续可以重写父类的get_queryset()pro_data = self.get_queryset()serializer_obj = self.get_serializer(instance=pro_data, many=True)return Response(serializer_obj.data, status=status.HTTP_200_OK)# 创建数据def post(self, request):serializer_obj = self.get_serializer(data=request.data)serializer_obj.is_valid(raise_exception=True)serializer_obj.save()return Response(serializer_obj.data, status=status.HTTP_201_CREATED)

 说明:

  1. 在上面代码中,代用父类的get_object()方法,不用再传递pk值,因为get_object()内部已经实现了处理查询集,lookup_field提取出所需要得pk值
  2. 在调用序列化器类传递data时,使用request.data获取到入参数据(继承APIView中的解析器)
  3. 在ProjectViews类方法中(get\post),往往不要直接使用queryset类属性,去获取查询集对象
    1. 一般需要使用self.get_queryset(),去获取查询集对象,因为更加灵活,后续可以重写父类的get_queryset()
    2. 调用self.get_serializer,获取序列化器类对象,原因同理

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/14201.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

03-MySQL-基础篇-SQL之DDL语句

SQL之DDL语句 前言DDL数据库操作表操作查询操作数据类型案例修改删除 前言 本篇来学习下SQL中的DDL语句 DDL 全称Data Definition Language,数据定义语言,用来定义数据库对象(数据库,表,字段) 数据库操作 查询所有数据库 sh…

【Qt QML入门】第一个Quick应用

运行结果: 打开Qt Creator,创建一个Qt Quick Qpplication,IDE为我们创建一个应用工程,其中包含如下文件: .pro工程文件,我们通过它来打开整个工程: QT quick# You can make your code fail to…

基于simulink识别彩色视频序列中的交通警告标志

一、前言 此示例演示如何识别彩色视频序列中的交通警告标志,如“停止”、“请勿进入”和“让行”。 二、模型 下图显示了交通警告标志识别模型: 三、交通警告标志模板 该示例使用两组模板 - 一组用于检测,另一组用于识别。 为了节省计算…

java的RSA加密解密示例

RSA算法是一种非对称加密算法,公钥和私钥都可以用于加密和解密操作。在RSA算法中,公钥用于加密数据,私钥用于解密数据。 具体来说,使用公钥加密的数据只能使用相应的私钥进行解密。而使用私钥加密的数据则可以使用相应的公钥进行…

【BUG历险记】vivado报错:binding vhdl entity ‘ xxxxx ‘does not have port ‘ xxxxx ‘解决方案

🏔【BUG历险记】vivado报错:binding vhdl entity xxxxx does not have port xxxxx 解决方案 问题描述 我在编写雷达的脉冲压缩仿真时,先例化了FFT和复数乘法器,并仿真验证通过了;再例化IFFT,结果仿真时…

最新抖音娱乐测评小程序源码 Thinkphp后端 抖音引流小程序

最新抖音娱乐测评小程序源码 thinkphp后端 抖音引流小程序 附搭建教程 测试环境 NginxPHP7.0MySQL5.6 网站运行目录设置为 /web 数据库配置文件 \source\application\database.php 后台登录地址 http://你的域名/index.php?s/admin/passport/login

Python学习笔记(十九)————json相关

目录 (1)什么是json (2) json有什么用 (3) json格式数据转化 (4)Python数据和Json数据的相互转化 (1)什么是json JSON 是一种轻量级的数据交互格式。可以按照…

ETHERNET/IP 转ETHERCAT连接ethercat总线伺服如何控制

远创智控YC-EIP-ECT网关连接到ETHERNET/IP总线中做为从站使用,连接到ETHERCAT总线中做为从站使用,可以同时满足多种工业生产的需求。支持广泛的设备类型,可以和多种不同的设备进行通讯。 技术参数 ETHERNET/IP 技术参数 ● 网关做为 ETHERN…

调戏chatGPT(三)逻辑推理

chatGPT已经被吹的神乎其神了,今天我来测试一下chatGPT的逻辑推理能力 t 你好 你好!很高兴能与您交流。 T 今天我想考考你的智力,看看你的逻辑思维能力如何 太好了!我非常乐意 t 一根质地不均匀的绳子完全烧完需要1小时…

Pytorch深度强化学习1-3:策略评估与贝尔曼期望方程详细推导

目录 0 专栏介绍1 从一个例子出发2 回报与奖赏3 策略评估函数4 贝尔曼期望方程5 收敛性证明 0 专栏介绍 本专栏重点介绍强化学习技术的数学原理,并且采用Pytorch框架对常见的强化学习算法、案例进行实现,帮助读者理解并快速上手开发。同时,辅…

基于springboot,vue网上订餐系统

开发工具:IDEA 服务器:Tomcat9.0, jdk1.8 项目构建:maven 数据库:mysql5.7 前端技术 :VueElementUI 服务端技术:springbootmybatisredis 本系统分用户前台和管理后台两部分,项…

P3804 【模板】后缀自动机(SAM)

题目描述 给定一个只包含小写字母的字符串 S。 请你求出 S 的所有出现次数不为 11 的子串的出现次数乘上该子串长度的最大值。 输入格式 一行一个仅包含小写字母的字符串 S。 输出格式 一个整数,为所求答案。 题解:这里就不讲后缀自动机的模板相关…