我有一个 PHP 脚本需要使用 HTTP 响应代码(状态代码)做出响应,例如 HTTP 200 OK,或者一些 4XX 或 5XX 代码。
我怎样才能在 PHP 中做到这一点?
我有一个 PHP 脚本需要使用 HTTP 响应代码(状态代码)做出响应,例如 HTTP 200 OK,或者一些 4XX 或 5XX 代码。
我怎样才能在 PHP 中做到这一点?
我刚刚发现了这个问题,并认为它需要一个更全面的答案:
从PHP 5.4开始,有三种方法可以实现这一点:
该header()
函数有一个特殊的用例,可以检测 HTTP 响应行并让您将其替换为自定义响应行
header("HTTP/1.1 200 OK");
但是,这需要对(快速)CGI PHP 进行特殊处理:
$sapi_type = php_sapi_name();
if (substr($sapi_type, 0, 3) == 'cgi')
header("Status: 404 Not Found");
else
header("HTTP/1.1 404 Not Found");
注意:根据HTTP RFC,原因短语可以是任何自定义字符串(符合标准),但为了客户端兼容性,我不建议在其中放置随机字符串。
注意: php_sapi_name()
需要PHP 4.0.1
使用第一个变体时显然存在一些问题。我认为最大的原因是它部分由 PHP 或 Web 服务器解析,并且文档记录不佳。
从 4.3 开始,该header
函数具有第三个参数,可以让您轻松地设置响应代码,但使用它需要第一个参数是非空字符串。这里有两个选项:
header(':', true, 404);
header('X-PHP-Response-Code: 404', true, 404);
我推荐第二个。第一个确实适用于我测试过的所有浏览器,但一些小型浏览器或网络爬虫可能会遇到仅包含冒号的标题行的问题。头字段名称在第 2 位。变体当然没有以任何方式标准化并且可以修改,我只是选择了一个有希望的描述性名称。
该http_response_code()
函数是在 PHP 5.4 中引入的,它使事情变得容易多了。
http_response_code(404);
就这样。
这是我在需要低于 5.4 的兼容性但想要“新”功能的功能时编写的一个http_response_code
功能。我相信 PHP 4.3 的向后兼容性绰绰有余,但你永远不知道......
// For 4.3.0 <= PHP <= 5.4.0
if (!function_exists('http_response_code'))
{
function http_response_code($newcode = NULL)
{
static $code = 200;
if($newcode !== NULL)
{
header('X-PHP-Response-Code: '.$newcode, true, $newcode);
if(!headers_sent())
$code = $newcode;
}
return $code;
}
}
不幸的是,我发现@dualed 提出的解决方案存在各种缺陷。
使用substr($sapi_type, 0, 3) == 'cgi'
不足以检测快速 CGI。使用 PHP-FPM FastCGI 进程管理器时,php_sapi_name()
返回 fpm 而不是 cgi
Fasctcgi 和 php-fpm 暴露了@Josh 提到的另一个错误 -header('X-PHP-Response-Code: 404', true, 404);
在 PHP-FPM (FastCGI) 下使用确实可以正常工作
header("HTTP/1.1 404 Not Found");
当协议不是 HTTP/1.1(即“HTTP/1.0”)时可能会失败。当前协议必须使用$_SERVER['SERVER_PROTOCOL']
(PHP 4.1.0 起可用
http_response_code()
调用导致意外行为至少有两种情况:
供您参考这里有 HTTP 响应状态代码的完整列表(此列表包括来自 IETF 互联网标准以及其他 IETF RFC 的代码。其中许多当前不受 PHP http_response_code 函数支持):http://en.wikipedia .org/wiki/List_of_HTTP_status_codes
您可以通过调用以下命令轻松测试此错误:
http_response_code(521);
如果您有例如自定义客户端应用程序调用您的服务器并期望一些额外的 HTTP 代码,服务器将发送“500 内部服务器错误”HTTP 响应代码导致意外错误。
我的解决方案(适用于自 4.1.0 以来的所有 PHP 版本):
$httpStatusCode = 521;
$httpStatusMsg = 'Web server is down';
$phpSapiName = substr(php_sapi_name(), 0, 3);
if ($phpSapiName == 'cgi' || $phpSapiName == 'fpm') {
header('Status: '.$httpStatusCode.' '.$httpStatusMsg);
} else {
$protocol = isset($_SERVER['SERVER_PROTOCOL']) ? $_SERVER['SERVER_PROTOCOL'] : 'HTTP/1.0';
header($protocol.' '.$httpStatusCode.' '.$httpStatusMsg);
}
结论
http_response_code() 实现不支持所有 HTTP 响应代码,并且可能会用同一组中的另一个响应代码覆盖指定的 HTTP 响应代码。
新的 http_response_code() 函数并不能解决所涉及的所有问题,但会导致最糟糕的情况是引入新的错误。
@dualed 提供的“兼容性”解决方案不能按预期工作,至少在 PHP-FPM 下是这样。
@dualed 提供的其他解决方案也有各种错误。快速 CGI 检测不处理 PHP-FPM。必须检测当前协议。
感谢任何测试和评论。
自 PHP 5.4 起,您可以使用http_response_code()
获取和设置标头状态代码。
这里有一个例子:
<?php
// Get the current response code and set a new one
var_dump(http_response_code(404));
// Get the new response code
var_dump(http_response_code());
?>
这是 php.net 中这个函数的文档:
如果您不使用输出缓冲,请在正文的任何输出之前添加此行。
header("HTTP/1.1 200 OK");
将消息部分 ('OK') 替换为适当的消息,并将状态代码替换为您的适当代码(404、501 等)
如果您在这里是因为 Wordpress 在加载环境时给出 404,这应该可以解决问题:
define('WP_USE_THEMES', false);
require('../wp-blog-header.php');
status_header( 200 );
//$wp_query->is_404=false; // if necessary
问题是由于它发送了一个 Status: 404 Not Found 标头。你必须覆盖它。这也将起作用:
define('WP_USE_THEMES', false);
require('../wp-blog-header.php');
header("HTTP/1.1 200 OK");
header("Status: 200 All rosy");
带头功能。在它采用的第一个参数部分中有一个示例。
如果您的 PHP 版本不包含此功能:
<?php
function http_response_code($code = NULL) {
if ($code !== NULL) {
switch ($code) {
case 100: $text = 'Continue';
break;
case 101: $text = 'Switching Protocols';
break;
case 200: $text = 'OK';
break;
case 201: $text = 'Created';
break;
case 202: $text = 'Accepted';
break;
case 203: $text = 'Non-Authoritative Information';
break;
case 204: $text = 'No Content';
break;
case 205: $text = 'Reset Content';
break;
case 206: $text = 'Partial Content';
break;
case 300: $text = 'Multiple Choices';
break;
case 301: $text = 'Moved Permanently';
break;
case 302: $text = 'Moved Temporarily';
break;
case 303: $text = 'See Other';
break;
case 304: $text = 'Not Modified';
break;
case 305: $text = 'Use Proxy';
break;
case 400: $text = 'Bad Request';
break;
case 401: $text = 'Unauthorized';
break;
case 402: $text = 'Payment Required';
break;
case 403: $text = 'Forbidden';
break;
case 404: $text = 'Not Found';
break;
case 405: $text = 'Method Not Allowed';
break;
case 406: $text = 'Not Acceptable';
break;
case 407: $text = 'Proxy Authentication Required';
break;
case 408: $text = 'Request Time-out';
break;
case 409: $text = 'Conflict';
break;
case 410: $text = 'Gone';
break;
case 411: $text = 'Length Required';
break;
case 412: $text = 'Precondition Failed';
break;
case 413: $text = 'Request Entity Too Large';
break;
case 414: $text = 'Request-URI Too Large';
break;
case 415: $text = 'Unsupported Media Type';
break;
case 500: $text = 'Internal Server Error';
break;
case 501: $text = 'Not Implemented';
break;
case 502: $text = 'Bad Gateway';
break;
case 503: $text = 'Service Unavailable';
break;
case 504: $text = 'Gateway Time-out';
break;
case 505: $text = 'HTTP Version not supported';
break;
default:
exit('Unknown http status code "' . htmlentities($code) . '"');
break;
}
$protocol = (isset($_SERVER['SERVER_PROTOCOL']) ? $_SERVER['SERVER_PROTOCOL'] : 'HTTP/1.0');
header($protocol . ' ' . $code . ' ' . $text);
$GLOBALS['http_response_code'] = $code;
} else {
$code = (isset($GLOBALS['http_response_code']) ? $GLOBALS['http_response_code'] : 200);
}
return $code;
}
header("HTTP/1.1 200 OK");
http_response_code(201);
header("Status: 200 All rosy");
http_response_code(200); 不起作用,因为测试警报 404 https://developers.google.com/speed/pagespeed/insights/