Django REST framework 学习笔记(二)
Request对象
- REST framework 引入了一个
Request对象,它是对HttpRequest的扩展,并提供灵活的请求解析 Request对象的核心功能是request.data,它类似于request.POST,但对于使用Web API更加有用
request.POST # 只能操作 form表单 数据,只能用于 'POST' 方法
request.data # 可以操作任意的数据,并且能用于 'POST'、'PUT'、'PATCH' 方法
Response对象
- REST framework 还引入了一个
Response对象,它是一种TemplateResponse对象 - 它能够使用协商好的类型,根据客户端请求的内容类型,呈现给客户端该类型的内容
- 例如:客户端请求
json类型的数据,则呈现给客户端json类型数据
return Response(data) # 根据客户端的请求呈现内容类型
状态码
- 使用数字HTTP状态码并不能总是正确的显示出读数,如果代码是错误的的话
- REST framework 为每个状态码提供了专门的标识符,放在了
status模块中 - 例如:
status.HTTP_400_BAD_REQUEST,表示状态码400
包装API视图
REST framework 提供了两个用于编写API视图的包装器
@api_view是处理基于函数方法的API视图的一个装饰器APIView是基于类的API视图
这些包装器提供了一些功能,例如:确保Request在视图中能接收到实例对象,以及向Response对象添加内容以呈现给客户端
并且提供了一些动作行为,例如:在适当的时候返回405 Method Not Allowed响应,还有当访问request.data出现输入格式错误时抛出ParseError异常
使用@api_view编写API视图
在上一篇文章中的项目案例中,修改 views.py 文件,我们不再使用JSONResponse,所以用下面的代码替换掉文件的内容
from django.http import HttpResponse
from rest_framework import status
from rest_framework.decorators import api_view
from rest_framework.response import Response
from .models import Snippet
from .serializers import SnippetSerializer
@api_view(['GET', 'POST'])
def snippet_list(request, format=None):
"""
列出所有代码段,或创建新代码段。
"""
if request.method == 'GET':
snippets = Snippet.objects.all()
serializer = SnippetSerializer(snippets, many=True)
return Response(serializer.data)
elif request.method == 'POST':
# data = JSONParser().parse(request)
serializer = SnippetSerializer(data=request.data)
if serializer.is_valid():
serializer.save()
return Response(serializer.data, status=status.HTTP_201_CREATED)
return Response(serializer.errors, status=status.HTTP_400_BAD_REQUEST)
@api_view(['GET', 'PUT', 'DELETE'])
def snippet_detail(request, pk, format=None):
"""
检索,更新或删除代码段。
"""
try:
snippet = Snippet.objects.get(pk=pk)
except Snippet.DoesNotExist:
return Response(status=status.HTTP_404_NOT_FOUND)
if request.method == 'GET':
serializer = SnippetSerializer(snippet)
return Response(serializer.data)
elif request.method == 'PUT':
# data = JSONParser().parse(request)
serializer = SnippetSerializer(snippet, data=request.data)
if serializer.is_valid():
serializer.save()
return Response(serializer.data)
return Response(serializer.errors, status=status.HTTP_400_BAD_REQUEST)
elif request.method == 'DELETE':
Snippet.delete()
return HttpResponse(status=status.HTTP_204_NO_CONTENT)
- 现在的代码看起来更加简洁一点,其中还使用了命名的状态码,使响应的意义更加明显
- 增加了
format参数用于指定请求的类型,默认为None,没有指定的时候,会返回它默认的视图 - 我们不再明确地将我们的请求或响应绑定到给定的内容类型,
request.data可以处理json请求,也可以处理其他格式的请求 - 再修改
snippets目录下的urls.py文件
from django.urls import path
from rest_framework.urlpatterns import format_suffix_patterns # 添加
from snippets import views
urlpatterns = [
path('snippets/', views.snippet_list),
path('snippets/<int:pk>', views.snippet_detail),
]
urlpatterns = format_suffix_patterns(urlpatterns) # 添加
- 进行测试:
http http://127.0.0.1:8000/snippets.json # 返回 JSON 格式
http http://127.0.0.1:8000/snippets.api # 返回 HTML API 格式
使用APIView编写视图
- APIView是基于类的API视图
- 修改
views.py文件
from snippets.models import Snippet
from snippets.serializers import SnippetSerializer
from django.http import Http404
from rest_framework.views import APIView
from rest_framework.response import Response
from rest_framework import status
class SnippetList(APIView):
"""
列出所有代码段,或创建新代码段。
"""
def get(self, request, format=None):
snippets = Snippet.objects.all()
serializer = SnippetSerializer(snippets, many=True)
return Response(serializer.data)
def post(self, request, format=None):
serializer = SnippetSerializer(data=request.data)
if serializer.is_valid():
serializer.save()
return Response(serializer.data, status=status.HTTP_201_CREATED)
return Response(serializer.errors, status=status.HTTP_400_BAD_REQUEST)
class SnippetDetail(APIView):
"""
检索,更新或删除代码段。
"""
def get_object(self, pk):
try:
return Snippet.objects.get(pk=pk)
except Snippet.DoesNotExist:
raise Http404
def get(self, request, pk, format=None):
snippet = self.get_object(pk)
serializer = SnippetSerializer(snippet)
return Response(serializer.data)
def put(self, request, pk, format=None):
snippet = self.get_object(pk)
serializer = SnippetSerializer(snippet, data=request.data)
if serializer.is_valid():
serializer.save()
return Response(serializer.data)
return Response(serializer.errors, status=status.HTTP_400_BAD_REQUEST)
def delete(self, request, pk, format=None):
snippet = self.get_object(pk)
snippet.delete()
return Response(status=status.HTTP_204_NO_CONTENT)
- 这与基于函数方法的API视图还是很相似
- 接下来修改
snippets目录下的urls.py文件
from django.urls import path
from rest_framework.urlpatterns import format_suffix_patterns
from snippets import views
urlpatterns = [
path('snippets/', views.SnippetList.as_view()),
path('snippets/<int:pk>/', views.SnippetDetail.as_view()),
]
urlpatterns = format_suffix_patterns(urlpatterns)
- 测试方法与
@api_view一样, 效果也是一致的。
本文详细介绍了Django REST framework中的Request对象、Response对象、状态码的使用,以及如何包装API视图和使用@api_view及APIView编写视图。通过具体的代码示例和修改过程,展示了如何创建简洁且明确的API接口。

被折叠的 条评论
为什么被折叠?



