如何实现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

相关推荐

  • 铁通宽带升级怎么弄?铁通宽带升级多少钱

    2026 年铁通宽带全面升级后,其核心优势在于“光网化改造完成”与“融合套餐性价比”,对于追求极致性价比及存量老用户,铁通宽带依然是优于部分一线运营商的优选,但需警惕部分区域“二次转售”带来的服务差异,随着 2026 年“双千兆”战略的深化落地,中国宽带市场格局发生微妙变化,铁通宽带(现多整合于中国移动体系)依……

    2026年5月9日
    0941
  • php网站打开时间长怎么办,php网站加载慢如何解决

    PHP网站打开速度慢的根本原因通常归结为服务器响应时间过长、代码执行效率低下以及数据库查询阻塞,解决这一问题的核心在于优化代码逻辑、升级服务器架构以及实施高效的缓存策略,对于企业级应用而言,速度不仅关乎用户体验,更直接影响SEO排名与业务转化,解决PHP网站加载缓慢的问题,不能仅靠单点优化,而需要从底层环境到应……

    2026年3月19日
    01203
  • Polardb数据库性能大赛,参赛者如何通过此大赛提升数据库性能优化能力?

    Polardb数据库性能大赛作为云原生数据库领域的权威性能验证平台,自2020年启动以来,已连续多年吸引国内外主流数据库厂商参与,旨在通过真实场景下的性能测试,客观评估各数据库产品的技术实力与应用价值,该大赛覆盖事务型、分析型、混合负载等多种业务场景,测试指标包括TPS(每秒事务数)、QPS(每秒查询数)、延迟……

    2026年1月8日
    02110
    • 服务器间歇性无响应是什么原因?如何排查解决?

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

      2026年1月10日
      020
  • 大模型API密钥泄露了怎么紧急处理,API密钥泄露紧急处理

    大模型API密钥一旦泄露,必须立即执行“停用旧密钥-生成新密钥-排查日志-重置权限”的四步紧急响应流程,以阻断潜在的数据滥用与算力盗用风险,在2026年人工智能应用深度普及的背景下,API密钥已不仅是访问凭证,更是企业数字资产的核心钥匙,根据中国信通院发布的《2026年大模型安全治理白皮书》显示,超过60%的大……

    2026年6月17日
    0503

发表回复

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