【DRF开发手册】使用 Django Rest Framework 的 @action 定义自定义方法

news2025/1/11 20:59:17

本文节选自笔者博客: https://www.blog.zeeland.cn/archives/so3f209hfeac

  • 💖 作者简介:大家好,我是Zeeland,全栈领域优质创作者。
  • 📝 CSDN主页:Zeeland🔥
  • 📣 我的博客:Zeeland
  • 📚 Github主页: Undertone0809 (Zeeland) (github.com)
  • 🎉 支持我:点赞👍+收藏⭐️+留言📝
  • 📣 系列专栏:django开发手册🍁
  • 💬介绍:The mixture of software dev+Iot+ml+anything🔥

在这里插入图片描述

Django系列专栏

  • 【django开发手册】Django 中使用自定义用户模型:一个比自带 User 更强的选择
  • 【django开发手册】如何使用select_related进行一次连表查询
  • 【django开发手册】drf通过添加自定义字段优化DRF序列化器,轻松实现高速API
  • 【django开发手册】解决admin添加外键下拉显示外键的问题
  • 【Django Rest Framework优化实践】ResponseResult、异常处理方法详解
  • 【DRF】深度分析枚举类型在DRF中的序列化问题及解决方案
  • 【django开发手册】关于django admin添加表信息的时候外键无法为空的问题解决方案
  • 【django开发手册】django admin如何显示外键对应的字段
  • 【django开发手册】DRF外键模型查询没有信息?教你实现序列化返回

【DRF开发手册】使用 Django Rest Framework 的 @action 定义自定义方法

前言

如果你正在使用 Django Rest Framework 来构建 RESTful API,那么你一定会经常使用到 ViewSet 这个工具类。ViewSet 提供了一些常见操作的映射,比如 list、create、retrieve、update、destroy 等,能够很方便地实现 CRUD 操作。

不过,如果你需要实现一些比较特殊的操作,那么 ViewSet 的默认方法就可能无法满足你的需求。例如,在一个投票系统中,我们需要提供一个对某个选项进行投票的接口,此时 ViewSet 默认的 list、create、retrieve 等方法就已经无法胜任此任务了。

在这种情况下,我们可以使用 Django Rest Framework 提供的 @action 装饰器来自定义方法。

如何使用 @action 定义自定义方法

使用 @action 定义自定义方法很简单,只需要在 ViewSet 中定义一个方法,并在该方法上加上 @action 装饰器,就可以将该方法转换为一个 API 接口。例如,我们可以在一个投票系统的 ViewSet 中定义一个投票接口:

from rest_framework.decorators import action
from rest_framework.response import Response

class PollViewSet(viewsets.ModelViewSet):
    queryset = Poll.objects.all()
    serializer_class = PollSerializer

    @action(methods=['post'], detail=True)
    def vote(self, request, pk=None):
        poll = self.get_object()
        option_id = request.data.get('option', None)
        if not option_id:
            return Response({'error': 'option is required'})
        try:
            option = poll.options.get(pk=option_id)
        except Option.DoesNotExist:
            return Response({'error': 'invalid option'})
        option.votes += 1
        option.save()
        return Response({'success': True})

该投票接口仅支持 POST 请求,它将会根据传入的 option 参数来对某个选项进行投票。这里需要注意的是,需要通过 self.get_object() 方法来获取当前的 Poll 对象,需要通过 request.data 来获取 POST 请求的参数。

在定义完自定义方法之后,需要将该方法添加到路由中,这可以通过 Django Rest Framework 提供的默认路由机制来实现:

from rest_framework import routers

router = routers.DefaultRouter()
router.register(r'polls', PollViewSet)

urlpatterns = [
    path('', include(router.urls)),
]

这里的 router.register() 方法将会自动根据 PollViewSet 中定义的方法来生成对应的路由,并将其添加到 urlpatterns 中。

对于 @action 定制方法,Django Rest Framework 会自动生成对应的 URL,也就是说,对于上面的投票例子,URL 是:

/polls/<pk>/vote/

其中 <pk> 对应的是具体的 Poll 对象的 primary key。例如,如果你的 Poll 对象的 primary key 是 3,那么可以这样访问:

/polls/3/vote/

在这个例子中,@action(detail=True) 会自动在 URL 中添加 pk 参数(代表 Poll 对象的 primary key),如果没有设置 detail=True,那么 URL 中就不会出现 pk 参数。由此可以看出,@action(detail=True)@action(detail=False) 的区别在于 URL 中是否需要添加 pk 参数。

至此,我们已经可以使用localhost:8000/polls/<id>/vote来访问这个接口了,如果你在其他定制化的时候不需要传入id参数,那么你可以配置detail=False,这样子就可以使用localhost:8000/polls/vote来访问这个接口了。

测试自定义方法

在实现完自定义方法之后,我们需要对其进行测试,以确保其可以正常工作。这可以通过 Django Rest Framework 提供的 APIClient 来实现。

from django.test import TestCase
from django.urls import reverse
from rest_framework import status
from rest_framework.test import APIClient
from polls.models import Poll, Option

class PollAPITestCase(TestCase):

    def setUp(self):
        self.client = APIClient()

    def test_vote(self):
        poll = Poll.objects.create(title='test poll')
        option = Option.objects.create(poll=poll, title='option1')

        url = reverse('polls-vote', args=[poll.id])
        data = {'option': option.id}
        response = self.client.post(url, data, format='json')

        self.assertEqual(response.status_code, status.HTTP_200_OK)

在这个示例测试中,我们首先使用 Poll.objects.create() 方法创建了一个 Poll 对象和一个 Option 对象,接着使用 Django Rest Framework 提供的 reverse() 函数根据 URL 模式的名称反向生成对应的 URL。最后,我们使用 APIClient 的 post() 方法向该 URL 发送 post 请求,并将选项的主键作为数据传递进去。最终,我们断言响应的 HTTP 状态码是否为 200,以此来判断测试是否通过。

reverse() 函数用来根据 URL 模式的名称生成对应的 URL,它接收一个参数,该参数是 URL 模式的名称,会将该名称反向映射为对应的 URL。例如,reverse('polls-vote', args=[poll.id]) 将会生成 URL /polls/<poll.id>/vote/。如果 URL 中含有不止一个占位符,则可以使用类似于 [poll.id, option.id] 的方式将多个占位符传递到 reverse() 函数中,最终生成具有多个参数的 URL。

总结

在本文中,我们介绍了如何使用 Django Rest Framework 的 @action 装饰器来定义自定义方法,以及如何使用 Django Rest Framework 的 APIClient 对自定义方法进行单元测试。通过本文,你应该已经掌握了 @action 装饰器的用法,以及如何进行单元测试。希望这篇文章能够帮助你更好地理解 Django Rest Framework。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

C++ Primer Plus(第6版) 全书重点学习笔记

目录 第10章 对象和类 10.1 过程性编程和面向对象编程 10.2 抽象和类 10.2.1 类简介 10.2.2 实现类成员函数 10.3 类的构造函数和析构函数 10.3.1 声明和定义构造函数 10.3.2 使用构造函数 10.3.3 默认构造函数 10.3.4 析构函数 10.4 this指针 10.5 对象数组 10.6 …

[长安杯 2021学生组]baigei

Index 前言介绍漏洞 利用思路利用过程一.编写交互函数二.填充Tcache Bin三.释放Tcache Bin四.获取Libc地址五.Tcache Bin Attack六.完整EXP&#xff1a; 前言 最近有点迷茫&#xff0c;开始放松自己了。 心态还不是很对&#xff0c;需要继续调整。 介绍 本题是一题经典的堆题…

Java学习笔记:内部类,静态内部类,匿名内部类

​这是本人学习的总结&#xff0c;主要学习资料如下 疯狂Java讲义第三版&#xff0c;李刚编&#xff0c;电子工业出版社出版 目录 1、内部类1.1、内部类简介1.2、内部类与外部类的关系和区别&#xff1a;1.3、内部类的语法 2、 非静态内部类3、静态内部类4、匿名内部类 1、内部…

“链引擎”入驻案例 | 每天超过35万条存证上链,长安链支撑链上价值流动

引言 长安链“链引擎”计划&#xff08;Powered by Chainmaker&#xff09;(简称&#xff1a;PBC计划)是由长安链生态联盟发起的一项应用赋能计划&#xff0c;旨在以长安链技术体系为核心支撑&#xff0c;汇聚产业各方力量&#xff0c;为应用方提供技术、品牌、生态等支持&…

Keil系列教程03_主窗口和工具栏详细说明

1写在前面 本文先让大家简单认识一下Keil的主窗口界面&#xff0c;然后再进一步认识Keil的文件、编译和调试工具栏。 Toolbars工具栏就是在菜单下面的两行快捷图标按钮&#xff0c;这些快捷按钮之所以在工具栏里面&#xff0c;在于它们使用的频率较高。比如保存按钮、编译按钮…

ChatGPT智能AI对话软件

ChatGPT智能AI的市场前景非常广阔&#xff0c;因为随着人工智能技术的不断发展和应用&#xff0c;人们对于智能AI对话系统的需求也越来越大。未来&#xff0c;智能AI对话系统将在各个领域得到广泛应用&#xff0c;例如智能客服、智能家居、自动驾驶等等&#xff0c;这些都有助于…

STM32 HAL库PID控制电机 第二章 TB6612FNG芯片驱动GB37-520电机

STM32 HAL库PID控制电机 第二章 TB6612FNG芯片驱动GB37-520电机(HAL库) 1 电路图 2 TB6612简介 TB6612是双驱动&#xff0c;可同时驱动两个电机 STBY&#xff1a;接单片机的IO口清零电机全部停止&#xff0c;置1通过AIN1 AIN2&#xff0c;BIN1&#xff0c;BIN2 来控制正反转…

linux下静态库和动态库的制作

一.静态库的制作 linux下库的命名规则&#xff1a;在linux下以libXXX.a为命名&#xff0c;lib&#xff08;library&#xff09;前缀是固定的&#xff0c;代表这个是库。接下来介绍静态库的制作流程。 1.1通过gcc编译获得.o文件 一般源程序经过预处理完成头文件和宏的展开&am…

运行时内存数据区之虚拟机栈——局部变量表

这篇内容十分重要,文字也很多,仔细阅读后,你必定有所收获! 基本内容 与程序计数器一样&#xff0c;Java虚拟机栈&#xff08;Java Virtual Machine Stack&#xff09;也是线程私有的&#xff0c;它的生命周期与线程相同。虚拟机栈描述的是Java方法执行的线程内存模型&#xf…

我想知道,就目前形势而言,学java好还是C++好?

前言 就现实点看看&#xff0c;可以对比现在Java和C的市场占有率&#xff0c;可以看到&#xff0c;到目前为止&#xff0c;Java在国内编程语言的市场仍然是占据着大头&#xff0c;在招聘当中Java的人数占有率仍然是遥遥领先于C&#xff0c;Java目前开阔的市场以及其巨大的岗位…

阿里,字节,拼多多,B站挨个面试一遍,你们猜哪个待遇最高?

我面试的是软件测试岗位&#xff0c;去年中旬的时候从原来的公司离职了&#xff0c;不是工作不好&#xff0c;而是公司发展速度太慢&#xff0c;自己干了几年&#xff0c;也没有太大的成长。以我目前的工作经验和实力&#xff0c;我认为准备一两个月&#xff0c;进大厂不是什么…

VS2022下载安装与基本使用(写C语言)

最近遇到一种问题&#xff0c;就是想要写一写C语言的代码&#xff0c;但是网页编辑器功能不全&#xff0c;GCC需要安装Liunx系统&#xff0c;VS又体量太大过于复杂&#xff0c;用keil又需要连接硬件&#xff0c;所以比较纠结。 工作中通常使用的是Keil&#xff0c;但是如果有时…

有记忆功能的动态通讯录

目录 前言1.进行文件操作的改造1.1contact.h的改造1.2contact.c的改造1.3test.c的改造 2.带文件操作的动态通讯录源码2.1contact.h2.2contact.c2.3test.c 总结 前言 前面我们一起学习的动态通讯录&#xff0c;一退出此程序联系人的信息就不见了&#xff1b;学习了文件操作操作后…

cocos creator 中使用web worker

1.应用场景&#xff1a;一些阻塞线程的方法可以放到worker里面去执行&#xff0c;不影响主线程&#xff0c;避免页面卡顿。 啊&#xff0c;有人就会说了&#xff0c;setTimeout不就可以了吗&#xff0c;还有什么async... JS本身就是单线程设计的&#xff0c;不管你是setTimeo…

EIGRP配置邻居关系详解

1.2 EIGRP 邻居关系 1.2.1 实验目的 通过 EIGRP 邻居建立的相关实验&#xff0c;学习到如何调整 EIGRP 的 HELLO 和 HOLD 时间&#xff0c;使用 被动接口阻止不必要的邻居关系&#xff0c;认证 EIGRP 邻居&#xff0c;静态邻居的配置以及哪些参数影响 EIGRP 邻居建立。 1.2.…

【数据库】Redis数据类型详解

目录一、5种基本数据类型1. String2. List3. Hash4. Set5. ZSet二、3种特殊类型1. Bitmap2. HyperLogLog3. Geospatial index一、5种基本数据类型 Redis 共有 5 种基本数据结构&#xff1a;String&#xff08;字符串&#xff09;、List&#xff08;列表&#xff09;、Set&#…

【CVPR轻量级网络】- 追求更高的FLOPS(FasterNet)

文章目录 题目&#xff1a;摘要1 介绍CNN中FLOPs的计算 2 相关工作3 PConv和FasterNet的设计3.1 偏卷积作为基本算子&#xff08;PConv)3.2 PConv后接PWConv3.3 FasterNet作为通用骨干 4实验 题目&#xff1a; Run, Don’t Walk: Chasing Higher FLOPS for Faster Neural Netw…

Android---Jetpack之Paging

目录 Paging 组件的意思 Paging 支持的架构类型 Paging 的工作原理 PositionalDataSource PagekeyedDataSource ItemKeyedDataSource BoundaryCallback Paging 组件的意思 分页加载是在应用程序开发过程中十分常见的需求&#xff0c;Paging 就是 Google 为了方便 Andr…

JAVA局域网监听软件的设计与开发

网络监听软件是提供给网络安全管理人员进行安全管理的工具&#xff0c;可以用来监视网络的状态、数据流动情况以及网络上传输的信息&#xff0c;以获取有用信息。作为黑客来说&#xff0c;通过网络监听可以获取其所需信息&#xff08;比如密码等&#xff09;&#xff1b;对黑客…

初中级Android工程师如何快速成长寻求突破

前言 写这篇文章的初衷是看到很多同学在一家公司工作了三五年&#xff0c;因为技术没有得到提升而随着年龄的增长导致不敢提出涨薪和跳槽找工作。希望这篇文章能够给这些还是初中级Android工程师的朋友一些启发。 快速成长 我们在向领导提出加薪申请或者是准备跳槽到更大的平…