PHP实现令牌桶限流Redis list列表 Lpush rpop 实现令牌桶 - 限流 PHP实例

本文涉及的产品
Redis 开源版,标准版 2GB
推荐场景:
搭建游戏排行榜
云数据库 Tair(兼容Redis),内存型 2GB
简介: PHP实现令牌桶限流Redis list列表 Lpush rpop 实现令牌桶 - 限流 PHP实例
本文环境 Windows10,PHP7.1,Redis6.0,Yii 2.0\

不懂的可以评论或联系我邮箱:owen@owenzhang.com\

著作权归OwenZhang所有。商业转载请联系OwenZhang获得授权,非商业转载请注明出处。

令牌桶限流介绍

令牌桶算法 (Token Bucket) 和 Leaky Bucket 效果一样但方向相反的算法,更加容易理解。\

随着时间流逝,系统会按恒定 1/QPS 时间间隔 (如果 QPS=100, 则间隔是 10ms) 往桶里加入 Token (想象和漏洞漏水相反,有个水龙头在不断的加水), 如果桶已经满了就不再加了。\

新请求来临时,会各自拿走一个 Token, 如果没有 Token 可拿了就阻塞或者拒绝服务.

* 令牌桶的另外一个好处是可以方便的改变速度。

* 一旦需要提高速率,则按需提高放入桶中的令牌的速率。

* 一般会定时 (比如 1000 毫秒) 往桶中增加一定数量的令牌,有些变种算法则实时的计算应该增加的令牌的数量.

Redis list lpush rpop 介绍

list

Redis列表是简单的字符串列表,按照插入顺序排序。你可以添加一个元素到列表的头部(左边)或者尾部(右边)

一个列表最多可以包含 232 - 1 个元素 (4294967295, 每个列表超过40亿个元素)。


redis 127.0.0.1:6379> lpush keyname owen

(integer) 1

redis 127.0.0.1:6379> lpush keyname zhang

(integer) 2

redis 127.0.0.1:6379> lpush keyname redis

(integer) 3

redis 127.0.0.1:6379> lrange keyname 0 10

​

1) "owen"

2) "zhang"

3) "redis"

**LPUSH** 将三个值插入了名为 **runoobkey** 的列表当中

lpush

lpush 将一个或多个值插入到列表头部。 如果 key 不存在,一个空列表会被创建并执行 LPUSH 操作。 当 key 存在但不是列表类型时,返回一个错误。


redis 127.0.0.1:6379> lpush keyname value1..

rpop

rpop 移除列表的最后一个元素,返回值为移除的元素。


redis 127.0.0.1:6379> rpop keyname 

令牌桶父类

步骤如下:

  1. 加入令牌

注意:需要加入定时任务,定时增加令牌数量

  1. 获取令牌

  1. 重设令牌桶,填满令牌


<?php

​

namespace common\components;

​

use Yii;

use yii\redis\Connection;

​

/\*\*

 \* 令牌桶 - 限流

 \*

 \* 令牌桶算法 (Token Bucket) 和 Leaky Bucket 效果一样但方向相反的算法,更加容易理解。

 \* 随着时间流逝,系统会按恒定 1/QPS 时间间隔 (如果 QPS=100, 则间隔是 10ms) 往桶里加入 Token (想象和漏洞漏水相反,有个水龙头在不断的加水), 如果桶已经满了就不再加了。

 \* 新请求来临时,会各自拿走一个 Token, 如果没有 Token 可拿了就阻塞或者拒绝服务.

 \*

 \* 令牌桶的另外一个好处是可以方便的改变速度。

 \* 一旦需要提高速率,则按需提高放入桶中的令牌的速率。

 \* 一般会定时 (比如 1000 毫秒) 往桶中增加一定数量的令牌,有些变种算法则实时的计算应该增加的令牌的数量.

 \*

 \* Class TrafficShape

 \* @package common\components

 \*/

class TrafficShape

{

    /\*\*

     \* 令牌桶

     \*

     \* @var string

     \*/

    public $container;

​

    /\*\*

     \* 最大令牌数

     \*

     \* @var int

     \*/

    public $max;

​

    /\*\*

     \* @var Connection

     \*/

    protected $redis;

​

    /\*\*

     \* TrafficShaper constructor.

     \* @param int $max

     \* @param string $containe

     \*/

    public function \_\_construct($max = 300, $container = 'container')

    {

        $this->redis = Yii::$app->redis;

        $this->max = $max;

        $this->container = $container;

    }

​

    /\*\*

     \* 加入令牌

     \*

     \* 注意:需要加入定时任务,定时增加令牌数量

     \*

     \* @param int $num 加入的令牌数量

     \* @return int 加入的数量

     \*/

    public function add($num = 0)

    {

        // 当前剩余令牌数

        $curnum = intval($this->redis->llen($this->container));

        // 最大令牌数

        $maxnum = intval($this->max);

        // 计算最大可加入的令牌数量,不能超过最大令牌数

        $num = $maxnum >= $curnum + $num ? $num : $maxnum - $curnum;

        // 加入令牌

        if ($num > 0) {

            $token = array\_fill(0, $num, 1);

            $this->redis->lpush($this->container, ...$token);

            return $num;

        }

​

        return 0;

    }

​

    /\*\*

     \* 获取令牌

     \*

     \* @return bool

     \*/

    public function get()

    {

        return $this->redis->rpop($this->container) ? true : false;

    }

​

    /\*\*

     \* 重设令牌桶,填满令牌

     \*/

    public function reset()

    {

        $this->redis->lrem($this->container, 0, $this->max);

        $this->add($this->max);

    }

}

调用实例

步骤如下:

  1. 添加令牌数量

注意:需要加入定时任务,定时增加令牌数量

  1. 验证令牌桶请求过快

一般会定时 (比如 1000 毫秒) 往桶中增加一定数量的令牌,有些变种算法则实时的计算应该增加的令牌的数量.

添加令牌


/\*\*

     \* 添加令牌数量

     \*/

    public function add()

    {

        // 默认最大添加数量为300

        $trafficShaper = new TrafficShaper(300);

        $trafficShaper->add(50);

    }

请求过快


$trafficShaper = new TrafficShaper();

        if (!$trafficShaper->get()) {

            throw new TooManyRequestsHttpException('请求过快');

        }

Buy me a cup of coffee :)

觉得对你有帮助,就给我打赏吧,谢谢!

Buy me a cup of coffee :)

https://www.owenzhang.com/wechat_reward.png

相关实践学习
基于Redis实现在线游戏积分排行榜
本场景将介绍如何基于Redis数据库实现在线游戏中的游戏玩家积分排行榜功能。
云数据库 Redis 版使用教程
云数据库Redis版是兼容Redis协议标准的、提供持久化的内存数据库服务,基于高可靠双机热备架构及可无缝扩展的集群架构,满足高读写性能场景及容量需弹性变配的业务需求。 产品详情:https://www.aliyun.com/product/kvstore &nbsp; &nbsp; ------------------------------------------------------------------------- 阿里云数据库体验:数据库上云实战 开发者云会免费提供一台带自建MySQL的源数据库&nbsp;ECS 实例和一台目标数据库&nbsp;RDS实例。跟着指引,您可以一步步实现将ECS自建数据库迁移到目标数据库RDS。 点击下方链接,领取免费ECS&amp;RDS资源,30分钟完成数据库上云实战!https://developer.aliyun.com/adc/scenario/51eefbd1894e42f6bb9acacadd3f9121?spm=a2c6h.13788135.J_3257954370.9.4ba85f24utseFl
目录
相关文章
|
1天前
|
NoSQL 关系型数据库 MySQL
Redis 列表(List)
10月更文挑战第16天
9 2
|
4天前
|
消息中间件 存储 监控
redis 的List类型 实现 排行榜
【10月更文挑战第8天】
20 2
|
1月前
|
消息中间件 存储 NoSQL
剖析 Redis List 消息队列的三种消费线程模型
Redis 列表(List)是一种简单的字符串列表,它的底层实现是一个双向链表。 生产环境,很多公司都将 Redis 列表应用于轻量级消息队列 。这篇文章,我们聊聊如何使用 List 命令实现消息队列的功能以及剖析消费者线程模型 。
84 20
剖析 Redis List 消息队列的三种消费线程模型
|
14天前
|
存储 分布式计算 NoSQL
大数据-40 Redis 类型集合 string list set sorted hash 指令列表 执行结果 附截图
大数据-40 Redis 类型集合 string list set sorted hash 指令列表 执行结果 附截图
21 3
|
16天前
|
缓存 NoSQL Redis
Redis命令:列表模糊删除详解
Redis命令:列表模糊删除详解
16 3
|
27天前
|
消息中间件 存储 NoSQL
4)深度解密 Redis 的列表(List)
4)深度解密 Redis 的列表(List)
20 1
|
1月前
|
存储 JSON NoSQL
redis基本数据结构(String,Hash,Set,List,SortedSet)【学习笔记】
这篇文章是关于Redis基本数据结构的学习笔记,包括了String、Hash、Set、List和SortedSet的介绍和常用命令。文章解释了每种数据结构的特点和使用场景,并通过命令示例演示了如何在Redis中操作这些数据结构。此外,还提供了一些练习示例,帮助读者更好地理解和应用这些数据结构。
redis基本数据结构(String,Hash,Set,List,SortedSet)【学习笔记】
|
17天前
|
存储 缓存 NoSQL
数据的存储--Redis缓存存储(一)
数据的存储--Redis缓存存储(一)
53 1
|
17天前
|
存储 缓存 NoSQL
数据的存储--Redis缓存存储(二)
数据的存储--Redis缓存存储(二)
33 2
数据的存储--Redis缓存存储(二)
|
13天前
|
消息中间件 缓存 NoSQL
Redis 是一个高性能的键值对存储系统,常用于缓存、消息队列和会话管理等场景。
【10月更文挑战第4天】Redis 是一个高性能的键值对存储系统,常用于缓存、消息队列和会话管理等场景。随着数据增长,有时需要将 Redis 数据导出以进行分析、备份或迁移。本文详细介绍几种导出方法:1)使用 Redis 命令与重定向;2)利用 Redis 的 RDB 和 AOF 持久化功能;3)借助第三方工具如 `redis-dump`。每种方法均附有示例代码,帮助你轻松完成数据导出任务。无论数据量大小,总有一款适合你。
50 6