我的编程空间,编程开发者的网络收藏夹
学习永远不晚

RESTful API批量操作的实现

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

RESTful API批量操作的实现

要解决的问题

RESTful API对于批量操作存在一定的缺陷。例如资源的删除接口:
DELETE /api/resourse/<id>/
如果我们要删除100条数据怎么搞?难道要调用100次接口吗?
比较容易想到的是下面两种方案:

  1. 用逗号分割放进url里:/api/resource/1,2,3...
  2. 将需要删除的资源的id放到请求体里面

对于方案1,由于浏览器对url的长度存在限制,如果操作的资源过多就无法实现。
对于方案2,这种处理方式存在一定的风险,因为根据RPC标准文档,DELETE的请求体在语义上没有意义,一些网关、代理、防火墙在收到DELETE请求后,会把请求的body直接剥离掉。

所以我参考https://www.npmjs.com/package/restful-api,将批量处理的操作名称和数据全部放到请求体里,统一使用POST请求发送:

POST /api/resource/batch/
    Body: {
                "method": "create",
                "data": [ { "name": "Mr.Bean" }, { "name": "Chaplin" }, { "name": "Jim Carrey" } ]
            }

POST /api/resource/batch/
    Body: {
                "method": "update",
                "data": { "1": { "name": "Mr.Bean" }, "2": { "name": "Chaplin" } }
            }

POST /api/resource/batch/
    Body: {
                "method": "delete",
                "data": [1, 2, 3]
            }

Python实现

环境:python==3.6.5, django==2.2, djangorestframework==3.9.4

GenericViewSet中加入了一些自定义的分发逻辑,将相应的Batch View放在Mixin里实现可重用。

class BatchGenericViewSet(GenericViewSet):
    batch_method_names = ('create', 'update', 'delete')
    def batch_method_not_allowed(self, request, *args, **kwargs):
        method = request.batch_method
        raise exceptions.MethodNotAllowed(method, detail=f'Batch Method {method.upper()} not allowed.')
        
    def initialize_request(self, request, *args, **kwargs):
        request = super().initialize_request(request, *args, **kwargs)
        # 将batch_method从请求体中提取出来,方便后面使用 
        batch_method = request.data.get('method', None)
        if batch_method is not None:
            request.batch_method = batch_method.lower()
        else:
            request.batch_method = None
        return request
        
    def dispatch(self, request, *args, **kwargs):
           self.args = args
           self.kwargs = kwargs
           request = self.initialize_request(request, *args, **kwargs)
           self.request = request
           self.headers = self.default_response_headers
           try:
                self.initial(request, *args, **kwargs)
                # 首先识别batch_method并进行分发
                if request.batch_method in self.batch_method_names:
                    method_name = 'batch_' + request.batch_method.lower()
                    handler = getattr(self, method_name, self.batch_method_not_allowed)
                elif request.method.lower() in self.http_method_names:
                handler = getattr(self, request.method.lower(), self.http_method_not_allowed)
                else:
                    handler = self.http_method_not_allowed
                    
                response = handler(request, *args, **kwargs)
            except Exception as exc:
                response = self.handle_exception(exc)
            
            self.response = self.finalize_response(request, response, *args, **kwargs)
            return self.response

下面是Mixin,因为懒所以放在了一个里面:

class BatchMixin:

    def batch_create(self, request, *args, **kwargs):
        """
        Create a batch of model instance

        request body like this:
        {
            "method": "create",
            "data": [
                        {
                            "name": "Mr.Liu",
                            "age": 27
                        },
                        {
                            "name": "Chaplin",
                            "age": 88
                        }
                    ]
        }
        """
        data = request.data.get('data', None)
        if not isinstance(data, list):
            raise exceptions.ValidationError({'data': 'Data must be a list.'})
        serializer = self.get_serializer(data=data, many=True)
        serializer.is_valid(raise_exception=True)
        with transaction.atomic():
            self.perform_create(serializer)
        headers = self.get_success_headers(serializer.data)
        return Response(serializer.data, status=status.HTTP_201_CREATED, headers=headers)

    def batch_update(self, request, *args, **kwargs):
        """
        Update a batch of model instance

        request body like this:
        {
            "method": "update",
            "data": {
                        1: { "name": "Mr.Liu" },
                        2: { "name": "Jim Carrey" }
                    }
        }
        """
        data = request.data.get('data', None)
        if not isinstance(data, dict):
            raise exceptions.ValidationError({'data': 'Data must be a object.'})
        ids = [int(id) for id in data]
        queryset = self.get_queryset().filter(id__in=ids)
        results = []
        for obj in queryset:
            serializer = self.get_serializer(obj, data=data[str(obj.id)], partial=True)
            serializer.is_valid(raise_exception=True)
            with transaction.atomic():
                self.perform_update(serializer)
            results.append(serializer.data)
        return Response(results)

    def batch_delete(self, request, *args, **kwargs):
        """
        Delete a batch of model instance

        request body like this:
        {
            "method": "delete",
            "data": [1, 2]
        }
        """
        data = request.data.get('data', None)
        if not isinstance(data, list):
            raise exceptions.ValidationError({'data': 'Data must be a list.'})
        queryset = self.get_queryset().filter(id__in=data)
        with transaction.atomic():
            self.perform_destroy(queryset)
        return Response(status=status.HTTP_204_NO_CONTENT)

这样实现对于restframework框架的ModelPermission权限判定会出现问题,因为所有请求都是通过POST实现的,默认情况下无法对Model的增、删、改权限进行有效的判断。稍微修改下DjangoModelPermissions就可以了:

class BatchModelPermissions(DjangoModelPermissions):
    batch_method_map = {
        'create': 'POST',
        'update': 'PATCH',
        'delete': 'DELETE'
    }

    def has_permission(self, request, view):
        if getattr(view, '_ignore_model_permissions', False):
            return True

        if not request.user or (
                not request.user.is_authenticated and self.authenticated_users_only):
            return False

        queryset = self._queryset(view)
        # 这里,这里
        batch_method = getattr(request, 'batch_method', None)
        if batch_method is not None:
            perms = self.get_required_permissions(self.batch_method_map[batch_method], queryset.model)
        else:
            perms = self.get_required_permissions(request.method, queryset.model)

        return request.user.has_perms(perms)

参考:https://www.npmjs.com/package/restful-api

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

RESTful API批量操作的实现

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

RESTful API批量操作的实现

要解决的问题RESTful API对于批量操作存在一定的缺陷。例如资源的删除接口:DELETE /api/resourse//如果我们要删除100条数据怎么搞?难道要调用100次接口吗?比较容易想到的是下面两种方案:用逗号分割放进
2023-01-31

如何构建 Golang RESTful API,并实现 CRUD 操作?

通过创建 golang 项目并安装必要的包,我们可以构建一个功能齐全的 restful api。它使用 mysql 数据库进行 crud 操作:1. 创建和连接数据库;2. 定义数据结构;3. 设置路由并处理 http 请求;4. 实现 c
如何构建 Golang RESTful API,并实现 CRUD 操作?
2024-05-14

Python 批量操作设备的实现步骤

目录背景需求- 方案一- 方案二方案三准备工作动手操作总结全民编程的时代,怎么能少了测试同学的身影。最近在学习Python,刚好活学活用,来实战一波背景需求某天午休未睡醒的我,突然接到一个Leader在群里@小盆,把平台上100台设备硬盘都
2022-06-02

Python 批量操作设备的实现方法

这篇文章主要讲解了“Python 批量操作设备的实现方法”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Python 批量操作设备的实现方法”吧!目录背景需求- 方案一- 方案二方案三准备工作
2023-06-20

使用Node.js实现RESTful API的示例

RESTful基础概念 REST(Representational State Transfer)描述了一个架构样式的网络系统,它首次出现在 2000 年 Roy Fielding 的博士论文中。在REST服务中,应用程序状态和功能可以分为
2022-06-04

SpringMVC Restful api接口实现的代码

【前言】面向资源的 Restful 风格的 api 接口本着简洁,资源,便于扩展,便于理解等等各项优势,在如今的系统服务中越来越受欢迎。.net平台有WebAPi项目是专门用来实现Restful api的,其良好的系统封装,简洁优雅的代码实
2023-05-31

Android实现便于批量操作可多选的图片ListView实例

本文实例讲述了Android实现便于批量操作可多选的图片ListView。分享给大家供大家参考,具体如下: 之前项目需要实现一个可多选的图片列表,用户选中一到多张图片后,批量上传。但是网上有可多选普通列表的代码、也有单纯图片列表的代码,却没
2022-06-06

MyBatis ORM的批量操作优化

MyBatis ORM的批量操作优化是提高数据库操作效率的关键。通过减少数据库访问次数、降低网络开销、提高事务效率、减少资源消耗、优化内存使用、提高并发度以及降低锁的竞争,可以显著提升整体性能。以下是一些关于MyBatis ORM批量操作优
MyBatis ORM的批量操作优化
2024-09-16

golang如何实现一个restful微服务的操作

这篇文章将为大家详细讲解有关golang如何实现一个restful微服务的操作,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。什么是golanggolang 是Google开发的一种静态强类型、编译型、并发
2023-06-14

变量在Oracle批量操作中的效率

在Oracle数据库中,批量操作可以显著提高性能,特别是在处理大量数据时。使用批量操作时,变量的使用可以提高效率,因为它们可以减少与数据库的通信次数,从而减少网络开销。以下是关于在Oracle批量操作中使用变量的一些建议:使用批量绑定变量:
变量在Oracle批量操作中的效率
2024-08-28

Elasticsearch文档批量操作的方法

本篇内容介绍了“Elasticsearch文档批量操作的方法”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!Elasticsearch的文档操
2023-06-30

PHP高级特性:RESTful API的实现技巧

PHP 高级特性:RESTful API 的实现技巧RESTful API(Representational State Transfer)是一种设计风格,它遵循 REST 原则,允许客户端与服务器之间的无状态交互。本文将探讨 PHP 中
PHP高级特性:RESTful API的实现技巧
2024-05-15

实现一个完整的Node.js RESTful API的示例

前言 这篇文章算是对Building APIs with Node.js这本书的一个总结。用Node.js写接口对我来说是很有用的,比如在项目初始阶段,可以快速的模拟网络请求。正因为它用js写的,跟iOS直接的联系也比其他语言写的后台更加接
2022-06-04

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录