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

相关推荐

  • 怎么查询家里的宽带,宽带账号查询方法

    查询家庭宽带状态与套餐详情,最核心且高效的途径是优先联系宽带运营商客服,通过拨打官方热线、登录运营商官方 APP 或前往线下营业厅,即可实时获取带宽速率、剩余流量、到期时间及设备连接数等关键信息,对于追求技术极客体验的用户,直接登录光猫管理后台或路由器后台则是掌握网络底层数据、排查故障根源的终极手段,官方渠道查……

    2026年4月28日
    0523
  • 芒果tv宽带怎么办理?芒果tv宽带资费及办理方法

    2026 年芒果 TV 宽带并非独立物理线路,而是依托电信、联通、移动等运营商骨干网,通过“芒果 TV 会员 + 宽带”融合套餐形式提供的增值服务,其核心优势在于针对芒果生态内容的零流量加速与独家权益,适合重度芒果用户,但单论纯宽带性价比需对比当地运营商直签价格,2026 芒果 TV 宽带业务模式深度解析业务本……

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

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

      2026年1月10日
      020
  • 广电数字电视宽带怎么办理?广电数字电视宽带办理条件

    2026 年广电数字电视宽带凭借“电视 + 宽带”融合套餐的高性价比与低延迟特性,已成为家庭及中小商户在光纤普及后的最优替代方案,尤其适合对价格敏感且需兼顾视频业务的用户群体,2026 年广电宽带核心优势与行业地位随着 5G-A 与千兆光网建设的深入,广电网络(中国广电)已全面实现全国一张网,其宽带业务在 20……

    2026年5月2日
    0481
  • plsql链接服务器连接不上?解决配置错误的关键步骤是什么?

    PLSQL链接服务器是Oracle数据库系统提供的一种核心扩展功能,用于实现跨数据库、跨系统的数据访问与集成,它通过创建一个虚拟的“链接服务器”对象,定义外部数据源(如其他Oracle数据库、SQL Server、MySQL、文件系统等)的连接信息,使Oracle用户能够像操作本地表一样,通过SQL语句查询、更……

    2026年1月27日
    0800

发表回复

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