关掉cURL自动重定向需设置CURLOPT_FOLLOWLOCATION为false并禁用CURLOPT_MAXREDIRS;用CURLOPT_HEADER和CURLOPT_NOBODY捕获响应头,逐行解析Location字段,注意处理续行、空行及大小写;仅获取跳转目标时优先HEAD请求,失败则降级GET并设超时。

PHP cURL 默认会跟随重定向,怎么关掉?
cURL 默认开启 CURLOPT_FOLLOWLOCATION,遇到 301 / 302 响应时自动发起新请求,你根本看不到跳转前的响应头或原始 Location。要“跳过重定向”,本质是关掉这个开关,并手动处理跳转逻辑。
- 设
CURLOPT_FOLLOWLOCATION为false(注意:不是字符串"false") - 同时建议关掉
CURLOPT_MAXREDIRS(默认 20),避免意外触发递归限制报错 - 用
CURLOPT_HEADER或CURLOPT_NOBODY+CURLOPT_RETURNTRANSFER捕获响应头,从中提取Location
curl_setopt($ch, CURLOPT_FOLLOWLOCATION, false); curl_setopt($ch, CURLOPT_MAXREDIRS, 0); curl_setopt($ch, CURLOPT_HEADER, true); curl_setopt($ch, CURLOPT_NOBODY, true); curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
怎么从响应头里安全提取 Location?
不能直接用 stristr() 粗暴匹配,因为响应头可能含多行、大小写混杂、有空格缩进,甚至存在多个 Location:(比如中间代理注入)。推荐用 curl_getinfo($ch, CURLINFO_HEADER_OUT) 不够用,得靠 curl_exec() 返回的完整响应体拆分。
- 用
explode("\r\n\r\n", $response, 2)分离 header 和 body(即使CURLOPT_NOBODY开启,header 仍完整返回) - 对 header 部分逐行遍历,用
stripos($line, 'Location:')判断,再trim(substr($line, 9))取值 - 注意跳过空行、注释行(
#开头)和 continuation line(以空格/制表符开头的续行)
远程文件 URL 含重定向时,如何只取最终目标地址而不下载内容?
如果你只是想“知道它会跳去哪”,不用 curl_exec() 下载整个 body,节省带宽和时间。用 CURLOPT_NOBODY => true 发起 HEAD 请求即可,但要注意:部分服务对 HEAD 返回不一致(比如不返回 Location),此时需退化为 GET + CURLOPT_NOBODY => false + CURLOPT_TIMEOUT 限流。
- 优先试 HEAD:设
CURLOPT_NOBODY => true和CURLOPT_CUSTOMREQUEST => 'HEAD' - 若响应无
Location或返回 405,再切回 GET,但立即curl_setopt($ch, CURLOPT_TIMEOUT, 3)防卡死 - 别依赖
CURLINFO_HTTP_CODE判断是否跳转——301/302 是跳转信号,但也要看有没有Location头,有些服务返回 302 却没 Location(非法响应)
为什么 get_headers() 不可靠?
它底层调用 HTTP 扩展,行为受 allow_url_fopen 和 max_redirects php.ini 设置影响,且无法关闭重定向或自定义超时。更严重的是:它返回数组格式混乱,Location 键名大小写不统一(有时是 location,有时是 Location),还可能被中间 CDN 过滤掉。生产环境必须用 cURL 控制粒度。
立即学习“PHP免费学习笔记(深入)”;
-
get_headers($url, 1)的第二个参数为1才返回关联数组,否则是索引数组,Location位置不确定 - 即使开了
stream_context_set_default(['http' => ['max_redirects' => 0]]),某些 PHP 版本仍忽略该设置 - 无法获取原始响应状态码(如 301 还是 302),
get_headers()默认只返回最终成功状态码
http_build_url()(pecl_http)或手写补全逻辑。










