前往小程序,Get更优阅读体验!
立即前往
发布
社区首页 >专栏 >PHP版滑动时间窗口算法

PHP版滑动时间窗口算法

作者头像
码农编程进阶笔记
发布2021-12-04 09:54:40
发布2021-12-04 09:54:40
63100
代码可运行
举报
运行总次数:0
代码可运行
代码语言:javascript
代码运行次数:0
复制
<?php
/**
 * 作者:码农编程进阶笔记
 * 点赞、分享朋友圈是最大的支持
 */
session_start();
$time = 60;//60秒
$count = 10; //可访问 10次
//第一次初始化
if(!isset($_SESSION['count'])){
    $_SESSION['count'] = 1;
    $_SESSION['time'] = time();
    $_SESSION['cha'] = 0;
    $_SESSION['status'] = 'success';
    print_r($_SESSION);
}else {
    $now = time();
    $cha = $now - $_SESSION['time'];
    $avg = intval($time / $count);//平均多少秒可获得一次机会
    //如果超过次数
    if($_SESSION['count'] > $count){
        //如果时差超过平均可获得次数的时长
        if($cha > $avg){
            $_SESSION['count'] -= intval($cha / $avg); //计算可得多少次机会
            $_SESSION['count'] = max($_SESSION['count'], 0);//修正次数不能为负。
            $_SESSION['count']++;
            $_SESSION['time'] = $now;
            $_SESSION['cha'] = $cha;
            $_SESSION['status'] = 'success';
            print_r($_SESSION);
        }else{
            //如果时差没有超过 $avg ,则还是失败。
            $_SESSION['cha'] = $cha;
            $_SESSION['status'] = 'fail';
            print_r($_SESSION);
        }
    }else{
        //如果没超过次数正常访问
        if($cha > $avg) {
            $_SESSION['count'] -= intval($cha / $avg); //计算可得多少次机会
            $_SESSION['count'] = max($_SESSION['count'], 0);//修正次数不能为负。
        }
        $_SESSION['count']++;
        $_SESSION['time'] = $now;
        $_SESSION['cha'] = $cha;
        $_SESSION['status'] = 'success';
        print_r($_SESSION);
    }
}

如果要精确计算,则要记录每次访问以元素的形式记录时间戳,到数组,每次请求的时候,遍历数组元素中的时间戳,与当前时间比较,清理掉 N分钟之前的元素,然后再计算个数,如果个数没超,则允许,反之不行。

代码语言:javascript
代码运行次数:0
复制
/**
 * 滑动时间窗口
 * 每次成功访问时,记录访问时间点
 * 每次清理N分钟之前的访问时间点
 * 对访问次数进行计数,判断是否超过次数
 * 作者:码农编程进阶笔记
 * @param $minute
 * @param $count
 * @param $times
 * @return bool
 */
function timeWindow($minute, $count, &$times){

    $now = time();
    $point = $now - $minute * 60;//从当前时间往前推N分钟的时间点
    foreach($times as $key => $item){
        if($item < $point) unset($times[$key]); //把N分钟之前的访问清理掉
    }

    if(count($times) <= $count){
        $times[] = $now; //成功时,记录本次访问时间点
        return true;
    }
    return false;

}
本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2021-12-03,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 码农编程进阶笔记 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档