php關于$_SERVER中一些和環境有關的參數詳解

前面有一個關于$_SERVER的文檔,今天,來看看怎么不同環境下其攜帶參數的差異與統一。

  • REQUEST_URI
    由HTTP1.1協議定義,指向某個頁面的URI,去除開頭的協議、主機、端口等信息。如 http://www.digpage.com:8080/index.php/foo/bar?queryParams,REQUEST_URI/index.php/foo/bar?queryParams

  • X-REWRITE-URL
    當使用了以開啟ISAPI_Rewrite的IIS作為服務器時,ISAPI_Rewrite會在未對原始URI作任何修改前,將原始的REQUEST_URI以X-REWRITE0URL HTTP頭保存起來。

  • PATH_INFO
    CGI 1.1規范定義的環境變量。從形式上看。它是整個URI中,在腳本標識之后、查詢參數?之前的部分。對于Apache,需要設置AcceptPathinfo On,且在一個URL沒有</PATH_INFO>部分的時候 ,PATH_INFO無效。特殊情況,如http://www.digpage.com/index.php/,PATH_INFO為/。而對于 Nginx,則需要設置:
    fastcgi_split_path_info ^(.+?\.php)(/.*)$; fastcgi_param PATH_INFO $fastcgi_path_info;

  • ORIG_PATH_INFO
    指未經 PHP 處理過的原始的 PATH_INFO”。 這個在 Apache 和 Nginx 需要配置一番才行,但一般用不到,已經有 PATH_INFO 可以用了嘛。而在 IIS 中則有點怪, 對于 http://www.digpage.com/index.php/ ORIG_PATH_INFO/index.php/;對于 http://www.digapge.com/index.php ORIG_PATH_INFO/index.php 。

在yii2中的一個方法:用來獲取不同環境下統一的URI

<?php
//使用了ISAPI_Rewrite的IIS
if (isset($_SERVER['HTTP_X_REWRITE_URL'])) {
    $requestUri = $_SERVER['HTTP_X_REWRITE_URL'];
//一般情況,需要去掉 URL 中的協議、主機、端口等內容
} elseif (isset($_SERVER['REQUEST_URI'])) {
    $requestUri = $_SERVER['REQUEST_URI'];
    if ($requestUri !== '' && $requestUri[0] !== '/') {
        $requestUri = preg_replace('/^(http|https):\/\/[^\/]+/i', '', $requestUri);
    }
// IIS 5.0, PHP 以 CGI 方式運行,需要把查詢參數接上
} elseif (isset($_SERVER['ORIG_PATH_INFO'])) {
    $requestUri = $_SERVER['ORIG_PATH_INFO'];
    if (!empty($SERVER['QUERY_STRING'])) {
        $requestUri .= '?' . $SERVER['QUERY_STRING'];
    }
} else {
    throw new Exception('wrong');
}

echo $requestUri;
  • SCRIPT_FILENAME
    當前腳本的實際物理路徑,比如 /var/www/digpage.com/frontend/web/index.php , 或 WIN平臺的 D:\www\digpage.com\frontend\web\index.php 。 以 Nginx 為例,一般情況下,SCRIPT_FILENAME 有以下配置項:
fastcgi_split_path_info ^(.+?\.php)(/.*)$;
# 使用 document root 來得到物理路徑
fastcgi_param SCRIPT_FILENAME $document_root$fastcgi_script_name
  • SCRIPT_NAME
    CGI 1.1 規范所定義的環境變量,用于標識 CGI 腳本(而非腳本的輸出),如 http://www.digapge.com/path/index.php 中的/path/index.php。 仍以 Nginx 為例,SCRIPT_NAME 一般情況下有fastcgi_param SCRIPT_NAME $fastcgi_script_name的設置。絕大多數情況下,使用 SCRIPT_NAME 即可獲取當前腳本。

  • PHP_SELF
    PHP_SELF 是 PHP 自己實現的一個 $_SERVER 變量,是相對于文檔根目錄(documentroot)而言的。 對于 http://www.digpage.com/path/index.php?queryParams,PHP_SELF為 /path/index.php 。 一般 SCRIPT_NAME 與 PHP_SELF 無異。但是,在 PHP.INI 中,如 cgi.fix_pathinfo=1(默認即為 1)時, 對于形如http://www.digpage.com/path/index.php/post/view/123, 則PHP_SELF為 /path/index.php/post/view/123 。 而根據 CGI 1.1 規范,SCRIPT_NAME 僅為 /path/index.php ,至于剩余的 /post/view/123 則為 PATH_INFO。

  • ORIG_SCRIPT_NAME
    當 PHP 以 CGI 模式運行時,默認會對一些環境變量進行調整。 首當其沖的,就是SCRIPT_NAME 的內容會變成 php.cgi 等二進制文件,而不再是 CGI 腳本文件。 當然,設置 cgi.fix_pathinfo=0 可以關閉這一默認行為。但這導致的副作用比較大,影響范圍過大,不宜使用。 但天無絕人之路,九死之地總留一線生機,那就是 ORIG_SCRIPT_NAME,他保留了調整前 SCRIPT_NAME 的內容。 也就是說,在 CGI 模式下,可以使用ORIG_SCRIPT_NAME 來獲取想要的 SCRIPT_NAME。 請留意使用ORIG_SCRIPT_NAME 前一定要先確認它是否存在。

再來看看yii2中的相關方法:

<?php
$scriptFile = $_SERVER['SCRIPT_FILENAME'];
$scriptName = basename($scriptFile);

if (basename($_SERVER['SCRIPT_NAME']) === $scriptName){
    $_scriptUrl = $_SERVER['SCRIPT_NAME'];
} elseif (basename($_SERVER['PHP_SELF']) === $scriptName) {
    $_scriptUrl = $_SERVER['PHP_SELF'];
} elseif (isset($_SERVER['ORIG_SCRIPT_NAME']) && basename($_SERVER['ORIG_SCRIPT_NAME']) === $scriptName) {
    $_scriptUrl = $_SERVER['ORIG_SCRIPT_NAME'];
} elseif (!empty($_SERVER['DOCUMENT_ROOT']) && strpos($scriptFile, $_SERVER['DOCUMENT_ROOT']) === 0) {
    $_scriptUrl = str_replace("\\", '/', str_replace($_SERVER['DOCUMENT_ROOT'], '', $scriptFile));
} else {
    throw new Exception ('wrong');
}

$baseUrl = retirm(dirname($_scriptUrl), '\\/');

在yii中通過這些的配合,也有一個獲取pathinfo的方法,如下是邏輯代碼:

$pathinfo = $requestUri;

if (($pos = strpos($pathinfo , '?')) !== false) {
    $pathinfo = substr($pathinfo, 0, $pos);
}

if (!preg_match('%^(?:
    [\x09\x0A\x0D\x20-\x7E] # ASCII
    | [\xC2-\xDF][\x80-\xBF] # non-overlong 2-byte
    | \xE0[\xA0-\xBF][\x80-\xBF] # excluding overlongs
    | [\xE1-\xEC\xEE\xEF][\x80-\xBF]{2} # straight 3-byte
    | \xED[\x80-\x9F][\x80-\xBF] # excluding surrogates
    | \xF0[\x90-\xBF][\x80-\xBF]{2} # planes 1-3
    | [\xF1-\xF3][\x80-\xBF]{3} # planes 4-15
    | \xF4[\x80-\x8F][\x80-\xBF]{2} # plane 16
    )*$%xs', $pathinfo)
) {
    $pathInfo = utf8_encode($pathInfo);
}

if (strpos($pathinfo, $_scriptUrl) === 0) {
    $pathinfo = substr($pathinfo, strlen($_scriptUrl));
} elseif ($baseUrl === "" || strpos($pathinfo, $baseUrl) === 0) {
    $pathinfo = substr($pathinfo, strlen($baseUrl));
} elseif (isset($_SERVER['PHP_SELF']) && strpos($_SERVER['PHP_SELF'], $_scriptUrl) === 0) {
    $pathinfo = substr($_SERVER['PHP_SELF'], str($_scriptUrl));
} else {
    throw new Exception('wrong');
}
if ($pathinfo[0] === '/') {
    $pathinfo = substr($pathinfo, 1);
}

這樣,就獲得了當前請求的URI($requestUri)、腳本路徑($_scriptUrl)、pathinfo($pathinfo),這些變量都是一個健壯的框架所需利用的元素。

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。

推薦閱讀更多精彩內容