V2EX = way to explore
V2EX 是一个关于分享和探索的地方
Sign Up Now
For Existing Member  Sign In
yitalin
V2EX  ›  PHP

如何限制 PHP 每秒只执行一次?

  •  
  •   yitalin · Jul 16, 2022 · 5099 views
    This topic created in 1384 days ago, the information mentioned may be changed or developed.
    a.php 实际上每秒有 10 个 ip 同时访问,如何实现每秒只执行一个 ip 的请求,同一秒剩余 9 个的请求实际上代码是不执行的?
    想了半天找不到解题思路
    28 replies    2022-07-18 13:28:52 +08:00
    em70
        1
    em70  
       Jul 16, 2022
    修改 nginx 配置文件,把最大并发 IP 设为 1

    limit_conn perip 1
    ersic
        2
    ersic  
       Jul 16, 2022 via Android   ❤️ 2
    换个思路,让每个执行都持续一秒。
    dangyuluo
        3
    dangyuluo  
       Jul 16, 2022
    加锁和队列?
    xiangyuecn
        4
    xiangyuecn  
       Jul 16, 2022
    @em70 #1 @ersic #2 严禁掩耳盗铃🐶 😂
    realpg
        5
    realpg  
    PRO
       Jul 16, 2022
    这还用思路?

    首先,多个请求,他们得有一个协商的地方,就是得有个公共的地方能存储状态
    数据库,redis,memcache,或者 shmem
    然后按秒做标志,考虑好锁存机制
    或者利用关系数据库的一些特性去做,比如利用 mysql 的 unique 索引实现天然锁
    zjsxwc
        6
    zjsxwc  
       Jul 16, 2022 via Android
    用 mode 为 x 的 fopen 方式打开文件名为秒级时间戳的文件,
    这样再某一秒中只有第一个 fopen 能成功,

    https://www.php.net/manual/zh/function.fopen.php
    realpg
        7
    realpg  
    PRO
       Jul 16, 2022
    @zjsxwc #6
    然后第一个打开成功,30ms 后 PHP 执行退出了,下一个又打开了?
    imdong
        8
    imdong  
       Jul 16, 2022 via iPhone   ❤️ 1
    伪代码:
    if(!Redis::set(time())) return;
    zjsxwc
        9
    zjsxwc  
       Jul 16, 2022 via Android
    @realpg
    执行完 sleep 1 秒,
    嘿嘿
    treblex
        10
    treblex  
       Jul 16, 2022 via iPhone
    存一个时间戳,没过期之前都返回缓存结果,过期执行实际代码更新缓存
    ersic
        11
    ersic  
       Jul 16, 2022
    redis 存一个 1s 过期的 key ,存在就不执行,不存在就执行,执行完继续存这个 key 。
    k9982874
        12
    k9982874  
       Jul 16, 2022 via Android
    起消息队列,消费者一秒只处理一个
    vovov
        13
    vovov  
       Jul 16, 2022
    php 的话可以用 laravel 的原子锁

    use Illuminate\Support\Facades\Cache;

    $lock = Cache::lock('foo', 1);

    if ($lock->get()) {
    // 锁定 1 秒...
    }
    GGGG430
        14
    GGGG430  
       Jul 16, 2022
    public static function lock($key, $ttl)
    {
    return apcu_add($key, 1, $ttl);
    }
    misaka19000
        15
    misaka19000  
       Jul 16, 2022
    漏桶算法

    想要简单点可以直接用 apisix 网关管理
    wonderfulcxm
        16
    wonderfulcxm  
       Jul 16, 2022 via iPhone
    那不在这一秒执行的其他请求怎么处理,是直接返回 503 还是进入队列等待下一秒?
    cpstar
        17
    cpstar  
       Jul 16, 2022
    我觉得 LZ 没想清楚的事情是,不是一秒钟内执行一个,而是每一秒钟怎么调配给不同的请求,是时间分配给请求,而不是请求来抢占时间。
    keepeye
        18
    keepeye  
       Jul 16, 2022
    redis setnx ,过期时间设为 1 秒 setnx 返回 1 的那个执行,其他的不执行
    sutra
        19
    sutra  
       Jul 16, 2022
    搜“限流”,有算法,也有解决方案。
    pengtdyd
        20
    pengtdyd  
       Jul 16, 2022
    rxjs 可以做到,不过嘛这玩意学起来很难
    cszchen
        21
    cszchen  
       Jul 16, 2022 via iPhone
    是期望“每秒”只有一个请求,还是同一个时间点只有一个请求。区别很大,解决的思路很简单。
    装个 redis ,获取原子锁,没有群的要等
    liyunlong5
        22
    liyunlong5  
       Jul 17, 2022 via Android   ❤️ 1
    @em70 这个只是限制一个连接,真正限制请求要加上 limit req 1r/s
    rb6221
        23
    rb6221  
       Jul 17, 2022
    加一层缓存就行了呗,一秒内的请求都走缓存
    hoopan
        24
    hoopan  
       Jul 18, 2022
    php 进程池只开一个呢,为啥会有这种奇怪的需求,原始需求是啥?如果是防止并发出错,可以加锁。
    junwind
        25
    junwind  
       Jul 18, 2022
    第一个 ip 访问进来写一个 session ,第一个进来,如果有这个值就 return ,没试过,不知道能否实现你的要求
    xuelu520
        26
    xuelu520  
       Jul 18, 2022
    加锁就行了
    wolfie
        27
    wolfie  
       Jul 18, 2022
    java 思路
    请求扔队列,根据 ip 分组。
    所有请求共用互斥锁,定义个守护线程循环 wait 、notify.
    tianyou666shen
        28
    tianyou666shen  
       Jul 18, 2022
    About   ·   Help   ·   Advertise   ·   Blog   ·   API   ·   FAQ   ·   Solana   ·   822 Online   Highest 6679   ·     Select Language
    创意工作者们的社区
    World is powered by solitude
    VERSION: 3.9.8.5 · 75ms · UTC 20:18 · PVG 04:18 · LAX 13:18 · JFK 16:18
    ♥ Do have faith in what you're doing.