如何实现PHP限制单IP请求?完整方法与代码教程

在PHP中限制单IP请求频率是防止滥用和DDoS攻击的常见策略,以下是几种实现方法,根据需求选择适合的方案:

php限制单ip请求


方法1:基于Session计数(简单计数器)

<?php
session_start();
$ip = $_SERVER['REMOTE_ADDR'];
$limit = 10; // 允许的最大请求次数
$timeWindow = 60; // 时间窗口(秒)
// 初始化计数器
if (!isset($_SESSION['requests'])) {
    $_SESSION['requests'] = [
        'count' => 1,
        'start_time' => time()
    ];
} else {
    // 检查时间窗口
    if (time() - $_SESSION['requests']['start_time'] > $timeWindow) {
        // 重置计数器
        $_SESSION['requests'] = [
            'count' => 1,
            'start_time' => time()
        ];
    } else {
        // 增加计数
        $_SESSION['requests']['count']++;
        // 检查是否超过限制
        if ($_SESSION['requests']['count'] > $limit) {
            http_response_code(429); // Too Many Requests
            die("请求过于频繁,请稍后再试");
        }
    }
}
// 正常处理请求...
?>

缺点:Session基于Cookie,客户端禁用Cookie则失效。


方法2:使用文件存储(无Session依赖)

<?php
$ip = $_SERVER['REMOTE_ADDR'];
$limit = 10; // 最大请求次数
$timeWindow = 60; // 时间窗口(秒)
$logDir = __DIR__ . '/ip_logs/'; // 日志目录
// 确保目录存在
if (!is_dir($logDir)) mkdir($logDir, 0700, true);
$logFile = $logDir . md5($ip) . '.log';
// 读取现有记录
if (file_exists($logFile)) {
    $data = json_decode(file_get_contents($logFile), true);
    if (time() - $data['start_time'] > $timeWindow) {
        // 重置计数器
        $data = ['count' => 1, 'start_time' => time()];
    } else {
        $data['count']++;
        if ($data['count'] > $limit) {
            http_response_code(429);
            die("请求过多,请等待{$timeWindow}秒");
        }
    }
} else {
    $data = ['count' => 1, 'start_time' => time()];
}
// 保存记录
file_put_contents($logFile, json_encode($data));
?>

缺点:高并发时文件IO可能成为瓶颈。


方法3:使用Redis(高性能推荐)

<?php
$ip = $_SERVER['REMOTE_ADDR'];
$limit = 10; // 最大请求次数
$timeWindow = 60; // 时间窗口(秒)
$redis = new Redis();
$redis->connect('127.0.0.1', 6379); // 连接Redis
$key = "rate_limit:$ip";
// 使用Redis计数器
$current = $redis->incr($key);
if ($current === 1) {
    // 首次设置时添加过期时间
    $redis->expire($key, $timeWindow);
} elseif ($current > $limit) {
    http_response_code(429);
    die("请求过于频繁");
}
// 正常处理请求...
?>

优势:高性能,适合分布式环境。

php限制单ip请求


方法4:.htaccess限制(Apache专用)

在项目根目录创建.htaccess文件:

<IfModule mod_rewrite.c>
    RewriteEngine on
    RewriteMap access counter:/path/to/access_map.txt
    RewriteCond ${access:%{REMOTE_ADDR}} >10
    RewriteRule ^ - [F,L]
</IfModule>

需配合脚本更新计数器,适用于熟悉Apache配置的场景。


关键注意事项:

  1. 获取真实IP:如果使用代理(如CDN),需通过HTTP_X_FORWARDED_FOR获取真实IP:
    $ip = $_SERVER['HTTP_X_FORWARDED_FOR'] ?? $_SERVER['REMOTE_ADDR'];
  2. 分布式环境:使用Redis等共享存储,避免单点失效。
  3. 突发请求:考虑使用令牌桶算法(如Redis + Lua实现)。
  4. 白名单机制:对可信IP免除限制:
    $whitelist = ['192.168.1.1', '10.0.0.1'];
    if (in_array($ip, $whitelist)) return;

完整Redis令牌桶示例

<?php
$ip = $_SERVER['REMOTE_ADDR'];
$redis = new Redis();
$redis->connect('127.0.0.1', 6379);
$key = "token_bucket:$ip";
$capacity = 20;    // 桶容量
$refillRate = 10;  // 每秒补充10个令牌
// 使用Lua脚本保证原子性
$lua = <<<LUA
local key = KEYS[1]
local now = tonumber(ARGV[1])
local capacity = tonumber(ARGV[2])
local refillRate = tonumber(ARGV[3])
local lastTime = tonumber(redis.call('hget', key, 'time') or now)
local tokens = tonumber(redis.call('hget', key, 'tokens') or capacity)
-- 计算补充的令牌
local refill = math.floor((now - lastTime) * refillRate)
tokens = math.min(capacity, tokens + refill)
-- 检查令牌是否足够
if tokens < 1 then
    return 0
end
-- 消耗令牌并更新
redis.call('hmset', key, 'tokens', tokens-1, 'time', now)
redis.call('expire', key, math.ceil(capacity/refillRate)*2)
return 1
LUA;
$result = $redis->eval($lua, [$key, time(), $capacity, $refillRate], 1);
if (!$result) {
    http_response_code(429);
    die("请求超出速率限制");
}
?>

根据业务需求选择方案,对于高并发场景,Redis方案是最佳选择,结合Lua脚本可保证原子性和精确控制。

php限制单ip请求

图片来源于AI模型,如侵权请联系管理员。作者:酷小编,如若转载,请注明出处:https://www.kufanyun.com/ask/290319.html

(0)
上一篇 2026年2月10日 04:21
下一篇 2026年2月10日 04:26

相关推荐

  • Photoshop入门疑问,如何精确调整文字位置移动技巧揭秘?

    在Photoshop中挪动文字位置是一个基本的操作,可以帮助你调整文本布局以适应不同的设计需求,以下是一篇详细介绍如何在Photoshop中挪动文字位置的文章,Photoshop中挪动文字位置的方法使用文字工具步骤:打开Photoshop,创建或打开一个包含文字的文档,在工具栏中选择“文字工具”(T),点击文档……

    2025年12月20日
    01570
  • 虚拟主机怎么远程使用?支持哪些连接方式?

    在探讨虚拟主机的使用方式时,一个核心问题常常浮现:虚拟主机可以远程使用吗?答案是肯定的,虚拟主机从其设计理念之初,就是为了让用户能够跨越地理限制,通过互联网进行远程管理和维护,这里的“远程使用”与我们通常理解的“远程桌面”(如Windows的RDP)在概念和操作上存在显著差异,它并非指获得一个图形化的操作桌面……

    2025年10月26日
    0800
    • 服务器间歇性无响应是什么原因?如何排查解决?

      根源分析、排查逻辑与解决方案服务器间歇性无响应是IT运维中常见的复杂问题,指服务器在特定场景下(如高并发时段、特定操作触发时)出现短暂无响应、延迟或服务中断,而非持续性的宕机,这类问题对业务连续性、用户体验和系统稳定性构成直接威胁,需结合多维度因素深入排查与解决,常见原因分析:从硬件到软件的多维溯源服务器间歇性……

      2026年1月10日
      020
  • ps如何高效分解存储GIF图?揭秘技巧与步骤!

    在数字图像处理领域,Photoshop(简称PS)是一款功能强大的图像编辑软件,它不仅支持多种图像格式,还能对图像进行复杂的编辑和优化,GIF图作为一种常见的动态图像格式,在网页设计、社交媒体和动画制作中有着广泛的应用,本文将详细介绍如何在Photoshop中分解存储GIF图,以便更好地管理和编辑这些动态图像……

    2025年12月22日
    0900
  • PostgreSQL新建数据库是否比旧版更好?原因是什么?

    PostgreSQL作为一款功能强大、开源的关系型数据库管理系统,以其高度的可扩展性、丰富的数据类型、强大的事务处理能力及良好的社区支持而闻名,新建数据库时,PostgreSQL提供了灵活的配置选项和完善的工具链,能帮助用户快速搭建符合业务需求的数据库环境,相比其他数据库,其ACID兼容性、并发处理能力以及丰富……

    2025年12月27日
    0880

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注