thinkphp6: 用redis+lua做限流(php 8.1.1 / thinkphp v6.0.10LTS )

一,系统配置

.env 中配置redis

[REDIS0]
TYPE = redis
HOST = 127.0.0.1
PORT = 6379
PASSWORD =

说明:刘宏缔的架构森林是一个专注架构的博客,

网站:https://blog.imgtouch.com
本文: https://blog.imgtouch.com/index.php/2023/05/29/thinkphp6-yong-redis-lua-zuo-xian-liu-php-8-1-1-thinkphp-v6/

         对应的源码可以访问这里获取: https://github.com/liuhongdi/
         或: https://gitee.com/liuhongdi

说明:作者:刘宏缔 邮箱: 371125307@qq.com

二,编写php代码

1,controller/Goods.php
class Goods  extends BaseController
{
    /**
     * 商品详情
     *
     * @return \think\Response
     */
    public function Detail(){
 
        $rate = new RedisLuaRate();
        $res = $rate->setRate();
 
        $log = new BusinessLog("admin");
        $log->log("rate res:".$res.":");
 
        if ($res == "1") {
            return Result::Success("成功");
        } else {
            return Result::Error("1","超出访问频率限制");
        }
    }
}
说明:此处仅做演示,实际应用时应该把这个拦截限流的功能放到middleware中
 
2,lib/util/RedisLuaRate.php
<?php
namespace app\lib\util;
 
class RedisLuaRate {
    function setRate() {
        $lua = <<<SCRIPT
local key = KEYS[1];
local limit = tonumber(KEYS[2])
local length = tonumber(KEYS[3])
--redis.log(redis.LOG_NOTICE,' length: '..length)
local current = redis.call('GET', key)
if current == false then
   --redis.log(redis.LOG_NOTICE,key..' is nil ')
   redis.call('SET', key,1)
   redis.call('EXPIRE',key,length)
   --redis.log(redis.LOG_NOTICE,' set expire end')
   return '1'
else
   --redis.log(redis.LOG_NOTICE,key..' value: '..current)
   local num_current = tonumber(current)
   if num_current+1 > limit then
       return '0'
   else
       redis.call('INCRBY',key,1)
       return '1'
   end
end
SCRIPT;
 
        $redis = new \Redis();
        $redis->connect(env('redis0.host', '127.0.0.1'),env('redis0.port', '6379'));
        $redis->select(0);
        $key = "ip".request()->ip();
        $sequence = $redis->eval($lua, [$key,3,1], 3);
 
        $luaError = $redis->getLastError();
        if (isset($luaError)) {
           // print_r($luaError);
        }
 
        return $sequence;
    }
}

说明: eval函数的第3个参数为KEYS个数。 phpredis依据此值将KEYS和ARGV做区分

三,测试效果

1,用ab做并发测试:
liuhongdi@lhdpc:/data/php/admapi/app$ ab -c 10 -n 10 http://192.168.219.6:8000/goods/detail?goodsid=12323
This is ApacheBench, Version 2.3 <$Revision: 1879490 $>
Copyright 1996 Adam Twiss, Zeus Technology Ltd, http://www.zeustech.net/
Licensed to The Apache Software Foundation, http://www.apache.org/
 
Benchmarking 192.168.219.6 (be patient).....done
 
 
Server Software:        nginx/1.18.0
Server Hostname:        192.168.219.6
Server Port:            8000
 
Document Path:          /goods/detail?goodsid=12323
Document Length:        37 bytes
 
Concurrency Level:      10
Time taken for tests:   0.059 seconds
Complete requests:      10
Failed requests:        7
   (Connect: 0, Receive: 0, Length: 7, Exceptions: 0)
Total transferred:      2276 bytes
HTML transferred:       496 bytes
Requests per second:    169.79 [#/sec] (mean)
Time per request:       58.897 [ms] (mean)
Time per request:       5.890 [ms] (mean, across all concurrent requests)
Transfer rate:          37.74 [Kbytes/sec] received
 
Connection Times (ms)
              min  mean[+/-sd] median   max
Connect:        0    1   0.5      1       2
Processing:    12   26  11.0     25      44
Waiting:       11   26  11.2     25      44
Total:         13   27  10.9     26      45
 
Percentage of the requests served within a certain time (ms)
  50%     26
  66%     29
  75%     37
  80%     42
  90%     45
  95%     45
  98%     45
  99%     45
100%     45 (longest request)
2,查看代码在日志中写入的结果:
2022-01-24 16:37:50--192.168.219.6--rate res:1:
2022-01-24 16:37:50--192.168.219.6--rate res:1:
2022-01-24 16:37:50--192.168.219.6--rate res:0:
2022-01-24 16:37:50--192.168.219.6--rate res:0:
2022-01-24 16:37:50--192.168.219.6--rate res:1:
2022-01-24 16:37:50--192.168.219.6--rate res:0:
2022-01-24 16:37:50--192.168.219.6--rate res:0:
2022-01-24 16:37:50--192.168.219.6--rate res:0:
2022-01-24 16:37:50--192.168.219.6--rate res:0:
2022-01-24 16:37:50--192.168.219.6--rate res:0:

可以看到在同一秒钟内,有三次返回成功,其余是返回失败

四,查看php和thinkphp的版本 

php:
liuhongdi@lhdpc:/data/php/admapi$ php --version
PHP 8.1.1 (cli) (built: Dec 20 2021 16:12:16) (NTS)
Copyright (c) The PHP Group
Zend Engine v4.1.1, Copyright (c) Zend Technologies
    with Zend OPcache v8.1.1, Copyright (c), by Zend Technologies 
thinkphp:
liuhongdi@lhdpc:/var/www/html$ cd /data/php/admapi/
liuhongdi@lhdpc:/data/php/admapi$ php think version
v6.0.10LTS 

 

posted @ 2022-01-26 21:44  刘宏缔的架构森林  阅读(624)  评论(0编辑  收藏  举报