【Django-rest-framework框架】 第08回 自定义频率类,分页功能,排序功能,过滤功能

1. 自定义频率类

throttling.py

from rest_framework.throttling import BaseThrottle


class Custom(BaseThrottle):
    VISIT_RECORD = {}  # 存放用户访问记录{ip1:[时间1,时间2],ip2:[时间1,时间2],'192.168.1.101':[当前时间,]}

    def __init__(self):
        self.history = None

    def allow_request(self, request, view):
        # 在这里写逻辑:根据ip地址判断用户是不是超过了频率限制
        # (1)取出访问者ip
        ip = request.META.get('REMOTE_ADDR')
        import time
        ctime = time.time()  # 取出当前时间
        # (2)判断当前ip不在访问字典里,添加进去,并且直接返回True,表示第一次访问
        if ip not in self.VISIT_RECORD:
            self.VISIT_RECORD[ip] = [ctime, ]
            return True
        self.history = self.VISIT_RECORD.get(ip)  # 当前访问者的时间列表 [时间2,]
        # (3)循环判断当前ip的时间列表,有值,并且当前时间减去列表的最后一个时间大于60s,把这种数据pop掉,这样列表中只有60s以内的访问时间,
        while self.history and ctime - self.history[-1] > 60:  # 循环结束后,剩下的都是1分钟以后访问的时间
            self.history.pop()
        # (4)判断,当列表小于3,说明一分钟以内访问不足三次,把当前时间插入到列表第一个位置,返回True,顺利通过
        # (5)当大于等于3,说明一分钟内访问超过三次,返回False验证失败
        if len(self.history) < 3:
            self.history.insert(0, ctime)
            return True
        else:
            return False

    def wait(self):
        import time
        ctime = time.time()
        return 60 - (ctime - self.history[-1])

views.py

from rest_framework.views import APIView
from rest_framework.response import Response

from .throttling import Custom


class TestView(APIView):
    throttle_classes = [Custom, ]

    def get(self, request):
        return Response('ok')

2. 频率功能源码剖析

# SimpleRateThrottle
	-源码里执行的频率类的allow_request,读SimpleRateThrottle的allow_request
    
class SimpleRateThrottle(BaseThrottle):
    cache = default_cache
    timer = time.time
    cache_format = 'throttle_%(scope)s_%(ident)s'
    scope = None
    THROTTLE_RATES = api_settings.DEFAULT_THROTTLE_RATES
    def __init__(self):  # 只要类实例化得到对象就会执行,一执行,self.rate就有值了,而且self.num_requests和self.duration
        if not getattr(self, 'rate', None): # 去频率类中反射rate属性或方法,发现没有,返回了None,这个if判断就符合,执行下面的代码
            self.rate = self.get_rate()  #返回了  '3/m'
        #  self.num_requests=3
        #  self.duration=60
        self.num_requests, self.duration = self.parse_rate(self.rate)

    def get_rate(self):
         return self.THROTTLE_RATES[self.scope] # 字典取值,配置文件中咱们配置的字典{'ss': '3/m',},根据ss取到了 '3/m'

    def parse_rate(self, rate):
        if rate is None:
            return (None, None)
        # rate:字符串'3/m'  根据 / 切分,切成了 ['3','m']
        # num=3,period=m
        num, period = rate.split('/')
        # num_requests=3  数字3
        num_requests = int(num)
        # period='m'  ---->period[0]--->'m'
        # {'s': 1, 'm': 60, 'h': 3600, 'd': 86400}[period[0]]
        # duration=60
        duration = {'s': 1, 'm': 60, 'h': 3600, 'd': 86400}[period[0]]
        # 3     60
        return (num_requests, duration)

    def allow_request(self, request, view):
        if self.rate is None:
            return True
        # 咱们自己写的,返回什么就以什么做限制  咱们返回的是ip地址
        # self.key=当前访问者的ip地址
        self.key = self.get_cache_key(request, view)
        if self.key is None:
            return True
        # self.history 访问者的时间列表,从缓存中拿到,如果拿不到就是空列表,如果之前有 [时间2,时间1]
        self.history = self.cache.get(self.key, [])
        # 当前时间
        self.now = self.timer()
        while self.history and self.history[-1] <= self.now - self.duration:
            self.history.pop()
        if len(self.history) >= self.num_requests:
            return self.throttle_failure()
        return self.throttle_success()
    
    
# 总结:以后要再写频率类,只需要继承SimpleRateThrottle,重写get_cache_key,配置类属性scope,配置文件中配置一下就可以了

3. 分页功能

查询所有的接口才需要分页
分页后端写法是固定的,前端展现形式是不一样的
    pc端的下一页的点击
    app中,翻页是下拉加载更多
from rest_framework.pagination import PageNumberPagination, LimitOffsetPagination, CursorPagination

3.1 PageNumberPagination

pagination.py

from rest_framework.pagination import PageNumberPagination, LimitOffsetPagination, CursorPagination


class OutPageNumberPagination(PageNumberPagination):
    # 写四个类属性即可
    page_size = 2  # 每页显示的条数
    page_query_param = 'page'  # /books/?page=3  查询第几页的参数
    page_size_query_param = 'size'  # /books/?page=3&size=4 查询第三页,每页显示4条
    max_page_size = 5  # 限制通过size查询,最大的条数

views.py

from rest_framework.generics import ListAPIView
from .models import Book
from .serializer import BookSerializer
from rest_framework.mixins import CreateModelMixin
from rest_framework.viewsets import ViewSetMixin
from .pagination import OutPageNumberPagination


class BookView(ViewSetMixin, ListAPIView, CreateModelMixin):
    pagination_class = OutPageNumberPagination  # 必需继承GenericAPIView才有这个类属性
    queryset = Book.objects.all()
    serializer_class = BookSerializer


3.2 LimitOffsetPagination

class OutLimitOffsetPagination(LimitOffsetPagination):
    default_limit = 2  # 每页显示的条数
    limit_query_param = 'limit'  # /books/?limit=4 这一页显示4条数据
    offset_query_param = 'offset'  # /books/?offset=3&limit=4 # 从第三页开始,取4条数据
    max_limit = 5  # 限制最多多少条


3.3 CursorPagination

class OutCursorPagination(CursorPagination):  # 只能上一页和下一页,它的分页效率是最高的,高于上面所有的分页方式,大数据量的分页,建议使用这种
    cursor_query_param = 'cursor'
    page_size = 3  # 每页显示的条数
    ordering = 'id'  # 排序,必须是表中的字段


3.4 drf中分页的使用

写一个类,继承drf提供的三个分页类之一
重写某几个类属性
PageNumberPagination : 正常分页,可以查询第几页,每页查询多少条
LimitOffsetPagination :偏移分页,可以直接从某个位置取几条
CursorPagination : 游标分页,只能选择上一页或下一页
把它配置在继承自GenericAPIView+ListModelMixin的子视图类上
如果继承的是APIView,需要自己写
   page = MyPageNumberPagination()
   res = page.paginate_queryset(qs, request)

4. 排序功能

from rest_framework.filters import OrderingFilter


class BookView(ViewSetMixin, ListAPIView, CreateModelMixin):
    pagination_class = OutPageNumberPagination  # 必需继承GenericAPIView才有这个类属性
    queryset = Book.objects.all()
    serializer_class = BookSerializer
    # 排序类的配置
    filter_backends = [OrderingFilter, ]
    # 指定按那个字段排序
    ordering_fields = ['id', 'price']

必须是继承GenericAPIView+ListModelMixin的子视图类上
配置排序类:
    filter_backends=[OrderingFilter,]
配置排序的字段
    ordering_fields=['id','price']
支持前端的访问形式
    http://127.0.0.1:8000/books/?ordering=-price,id # 先按价格的降序排,如果价格一样再按id的升序排
    
纯自己写的,继承了APIView的,需要自己从请求地址中取出排序规则,自己排序
-'price','-id'=reqeust.query_params.get('ordering').split(',')
-qs = Book.objects.all().order_by('price','-id')
    
分页和排序能一起用,但是是先排序后分页的

5. 过滤功能

from rest_framework.filters import OrderingFilter, SearchFilter


class BookView(ViewSetMixin, ListAPIView, CreateModelMixin):
    pagination_class = OutPageNumberPagination  # 必需继承GenericAPIView才有这个类属性
    queryset = Book.objects.all()
    serializer_class = BookSerializer
    # 指定过滤类
    filter_backends = [SearchFilter, ]
    # 指定过滤字段
    search_fields = ['name']


内置过滤

# 查询所有才涉及到过滤,其它接口都不需要
# restful规范中有一条,请求地址中带过滤条件:分页,排序,过滤统称为过滤
# 使用内置过滤类使用步骤   查询所有才涉及到排序,其它接口都不需要
	必须是继承GenericAPIView+ListModelMixin的子视图类上
配置过滤类:
    filter_backends=[SearchFilter,]
配置过滤的字段
    ordering_fields=['name','publish']
支持前端的访问形式
    http://127.0.0.1:8000/books/?search=# 只要name中或publish中有三都能搜出来
内置过滤类只能通过search写条件,如果配置了多个过滤字段,是或者的条件
        
不够用:
    第三方:过滤类
    自己写:自己写过滤类

作业

1.自定义频率类

from rest_framework.throttling import BaseThrottle


class Custom(BaseThrottle):
    VISIT_RECORD = {}
    def __init__(self):
        self.history = None

    def allow_request(self, request, view):
        ip = request.META.get('REMOTE_ADDR')
        import time
        ctime = time.time()
        if ip not in self.VISIT_RECORD:
            self.VISIT_RECORD[ip] = [ctime,]
            return True
        self.history = self.VISIT_RECORD.get(ip)
        while self.history and ctime - self.history[-1] > 60:
            self.history.pop()
        if len(self.history) < 3:
            self.history.insert(0, ctime)
            return True
        else:
            return False
    
    def wait(self):
        import time
        ctime = time.time()
        return 60 - (ctime - self.history[-1])

2.使用3种分页方式,实现对查询所有数据接口的分页

from rest_framework.pagination import PageNumberPagination, LimitOffsetPagination, CursorPagination


class OutPageNumberPagination(PageNumberPagination):
    page_size = 6 
    page_query_param = 'page'  
    page_size_query_param = 'size'  
    max_page_size = 5  


class OutLimitOffsetPagination(LimitOffsetPagination):
    default_limit = 2 
    limit_query_param = 'limit' 
    offset_query_param = 'offset'  
    max_limit = 5  


class OutCursorPagination(CursorPagination):  
    cursor_query_param = 'cursor'
    page_size = 3  
    ordering = 'id'  

3.带排序,带按名字过滤

from rest_framework.generics import ListAPIView
from .models import Book
from .serializer import BookSerializer
from rest_framework.mixins import CreateModelMixin
from rest_framework.viewsets import ViewSetMixin
from .pagination import OutPageNumberPagination, OutLimitOffsetPagination, OutCursorPagination

from rest_framework.filters import OrderingFilter, SearchFilter


class BookView(ViewSetMixin, ListAPIView, CreateModelMixin):
    pagination_class = OutPageNumberPagination 
    queryset = Book.objects.all()
    serializer_class = BookSerializer

    filter_backends = [OrderingFilter,SearchFilter,  ]
    ordering_fields = ['id', 'price']
    search_fields = ['name','publish']

4 继承APIView实现分页

class BookView1(APIView):
    def get(self, request):
        qs = Book.objects.all()
        # 第一步:实例化得到分页类的对象
        page = OutPageNumberPagination()
        # 第二步:调用分页类对象的paginate_queryset方法,完成分页(传入了所有数据,传入了请求对象:apge=3和size=5)
        # 分页后的数据=qs[(page-1)*size:page*size]
        res = page.paginate_queryset(qs, request)
        ser = BookSerializer(instance=res, many=True)
        # res_data={'count':qs.count(),'next':'http://127.0.0.1/books/?page=4&size=5'}
        # return Response(ser.data)
        # 第三步:调用分页类对象的get_paginated_response方法返回
        return page.get_paginated_response(ser.data)
posted @   |相得益张|  阅读(44)  评论(0编辑  收藏  举报
相关博文:
阅读排行:
· 分享一个免费、快速、无限量使用的满血 DeepSeek R1 模型,支持深度思考和联网搜索!
· 基于 Docker 搭建 FRP 内网穿透开源项目(很简单哒)
· ollama系列01:轻松3步本地部署deepseek,普通电脑可用
· 25岁的心里话
· 按钮权限的设计及实现
点击右上角即可分享
微信分享提示