首页 > 其他分享 >【2022.10.10】drf(8)

【2022.10.10】drf(8)

时间:2022-10-10 22:13:10浏览次数:33  
标签:10 return self request rate 过滤 history 2022.10 drf

今日学习内容

  • 1.自定义评率类
  • 2.频率功能源码剖析
  • 3.分页功能
  • 4.排序功能
  • 5.过滤功能
    • 5.1内置过滤

1 自定义频率类

from rest_framework.throttling import BaseThrottle
class MyThrottle(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])

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中,翻页是下拉加载更多
# drf中分页的使用:
	-写一个类,继承drf提供的三个分页类之一
    -重写某几个类属性
    -把它配置在继承自GenericAPIView+ListModelMixin的子视图类上
    -如果继承的是APIView,需要自己写
    	page = MyPageNumberPagination()
        res = page.paginate_queryset(qs, request)

4 排序功能

# 查询所有才涉及到排序,其它接口都不需要

# 必须是继承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 过滤功能

5.1 内置过滤

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

作业

1 自定义频率类,写一遍
2 使用3种分页方式,实现对查询所有数据接口的分页
3 带排序,带按名字过滤

#----部分人写----
4 继承APIView,实现分页,返回格式跟之前一样
class BookView1(APIView):
    def get(self, request):
        qs = Book.objects.all()
        page = MyPageNumberPagination()
        res = page.paginate_queryset(qs, request)
        ser = BookSerializer(instance=res,many=True)
        return Response(ser.data)

标签:10,return,self,request,rate,过滤,history,2022.10,drf
From: https://www.cnblogs.com/55wym/p/16777607.html

相关文章

  • drf-频率类及分页排过滤序
    1自定义频率类取出访问者ip判断当前ip不在访问字典里,添加进去,并且直接返回True,表示第一次访问,在字典里,继续往下走循环判断当前ip的列表,有值,并且当前时间减去列表的最......
  • 玩转树莓派[10基地2.0开启9090端口和VNC内网穿透]
    title:玩转树莓派[10:基地2.0开启9090端口和VNC内网穿透]excerpt:内网穿透第二弹:可以远程操作桌面了~tags:[raspberry,基地2.0,9090端口,vnc,内网穿透,http协......
  • DRF
    DRF频率类#某个接口,限制访问频率----》可以根据IP,用户id#频率类的编写-第一步:写一个类,继承SimpleRateThrottle-第二步:重写get_cache_key方法-第三步:返......
  • 2022-10-10学习内容
    1.RelativeLayout1.1activity_relative_layout.xml<?xmlversion="1.0"encoding="utf-8"?><RelativeLayoutxmlns:android="http://schemas.android.com/apk/res/andr......
  • #yyds干货盘点#【愚公系列】2022年10月 微信小程序-全局配置属性之页面配置
    前言微信小程序是由一个个单页面组成,每个页面都是由固定规则的。小程序页面自身分为两个主要部分独立运行:view模块和service模块。在开发者工具中,它们独立运行于不同的......
  • drf 之分页,过滤,排序
    自定义频率类classMyThrottle(SimpleRateThrottle):VISIT_RECORD={}#存放用户访问记录{ip1:[时间1,时间2].。。}def__init__(self):self.hi......
  • 10 我可以不用发论文了?
    博客配套视频链接:https://www.bilibili.com/video/BV11g41127Zn/?spm_id_from=333.788&vd_source=b1ce52b6eb3a9e6c2360a4b7172edf5ab站直接看视频描述图片描述0:30......
  • Python 应用之求 100 以内的奇数和
    在数学中,我们需要用到很多求和的办法,比如说求1至100的和,还有100以内的所有偶数和和所有奇数和,如果我们慢慢地计算是不是很浪费时间,还容易出错。其实通过Python就可......
  • 【闲话】2022.10.10
    真的帮我妈登上了cnblogs好诶!今天Accoders考试T3T4真的不想改了除非我学到那一部分然后干了干李超树我完全明白了(不是继续搜索终于知道A*的用法了我好菜......
  • 20221005(补
    20221005题目简单点朴素算法​ 很容易想到存下每个位置后离它最近的各个字母的位置。然后再暴力地从答案区间\([l,r]\)的离左端点最近的\(e\)开始按\(e,a,s,y\)的顺序......