PHP中對(duì)用戶身份認(rèn)證實(shí)現(xiàn)兩種方法
更新時(shí)間:2011年06月04日 11:28:12 作者:
用戶在設(shè)計(jì)和維護(hù)站點(diǎn)的時(shí)候,經(jīng)常需要限制對(duì)某些重要文件或信息的訪問(wèn)。通常,我們可以采用內(nèi)置于WEB服務(wù)器的基于HTTP協(xié)議的用戶身份驗(yàn)證機(jī)制。
當(dāng)訪問(wèn)者瀏覽受保護(hù)頁(yè)面時(shí),客戶端瀏覽器會(huì)彈出對(duì)話窗口要求用戶輸入用戶名和密碼,對(duì)用戶的身份進(jìn)行驗(yàn)證,以決定用戶是否有權(quán)訪問(wèn)頁(yè)面。下面用兩種方法來(lái)說(shuō)明其實(shí)現(xiàn)原理。
一、用HTTP標(biāo)頭來(lái)實(shí)現(xiàn)
標(biāo)頭是服務(wù)器以HTTP協(xié)議傳送HTML信息到瀏覽器前所送出的字串。HTTP采用一種挑戰(zhàn)/響應(yīng)模式對(duì)試圖進(jìn)入受密碼保護(hù)區(qū)域的用戶進(jìn)行身份驗(yàn)證。具體來(lái)說(shuō),當(dāng)用戶首次向WEB服務(wù)器發(fā)出訪問(wèn)受保護(hù)區(qū)域的請(qǐng)求時(shí),挑戰(zhàn)進(jìn)程被啟動(dòng),服務(wù)器返回特殊的401標(biāo)頭,表明該用戶身份未經(jīng)驗(yàn)證??蛻舳藶g覽器在檢測(cè)到上述響應(yīng)之后自動(dòng)彈出對(duì)話框,要求用戶輸入用戶名和密碼。用戶完成輸入之后點(diǎn)擊確定,其身份識(shí)別信息就被傳送到服務(wù)端進(jìn)行驗(yàn)證。如果用戶輸入的用戶名和密碼有效,WEB服務(wù)器將允許用戶進(jìn)入受保護(hù)區(qū)域,并且在整個(gè)訪問(wèn)過(guò)程中保持其身份的有效性。相反,若用戶輸入的用戶名稱(chēng)或密碼無(wú)法通過(guò)驗(yàn)證,客戶端瀏覽器會(huì)不斷彈出輸入窗口要求用戶再次嘗試輸入正確的信息。整個(gè)過(guò)程將一直持續(xù)到用戶輸入正確的信息位置,也可以設(shè)定允許用戶進(jìn)行嘗試的最大次數(shù),超出時(shí)將自動(dòng)拒絕用戶的訪問(wèn)請(qǐng)求。
在PHP腳本中,使用函數(shù)header()直接給客戶端的瀏覽器發(fā)送HTTP標(biāo)頭,這樣在客戶端將會(huì)自動(dòng)彈出用戶名和密碼輸入窗口,來(lái)實(shí)現(xiàn)我們的身份認(rèn)證功能。在PHP中,客戶端用戶輸入的信息傳送到服務(wù)器之后自動(dòng)保存在 $PHP_AUTH_USER,$PHP_AUTH_PW,以及 $PHP_AUTH_TYPE這三個(gè)全局變量中。利用這三個(gè)變量,我們可以根據(jù)保存在數(shù)據(jù)文件或者數(shù)據(jù)庫(kù)中用戶帳號(hào)信息來(lái)驗(yàn)證用戶身份!
不過(guò),需要提醒使用者注意的是:只有在以模塊方式安裝的PHP中才能使用$PHP_AUTH_USER,$PHP_AUTH_PW,以及 $PHP_AUTH_TYPE這三個(gè)變量。如果用戶使用的是CGI模式的PHP則無(wú)法實(shí)現(xiàn)驗(yàn)證功能。在本節(jié)后附有PHP的模塊方式安裝方法。
下面我們用Mysql數(shù)據(jù)庫(kù)來(lái)存儲(chǔ)用戶的身份。我們需要從數(shù)據(jù)庫(kù)中提取每個(gè)帳號(hào)的用戶名和密碼以便與$PHP_AUTH_USER和$PHP_AUTH_PW變量進(jìn)行比較,判斷用戶的真實(shí)性。
首先,在MySql中建立一個(gè)存放用戶信息的數(shù)據(jù)庫(kù)
數(shù)據(jù)庫(kù)名為XinXiKu ,表名為user;表定義如下:
create table user(
ID INT(4) NOT NULL AUTO_INCREMENT,
name VARCHAR(8) NOT NULL,
password CHAR(8) NOT NULL,
PRIMARY KEY(ID)
)
說(shuō)明:
1、ID為一個(gè)序列號(hào),不為零而且自動(dòng)遞增,為主鍵;
2、name為用戶名,不能為空;
3、password為用戶密碼,不能為空;
以下是用戶驗(yàn)證文件login.php
//判斷用戶名是否設(shè)置
if(!isset($PHP_AUTH_USER))
{
header("WWW-Authenticate:Basic realm="身份驗(yàn)證功能"");
header("HTTP/1.0 401 Unauthorized");
echo "身份驗(yàn)證失敗,您無(wú)權(quán)共享網(wǎng)絡(luò)資源!";
exit();
}
/*連接數(shù)據(jù)庫(kù)*/
$db=mysql_connect("localhost","root","");
//選擇數(shù)據(jù)庫(kù)
mysql_select_db("XinXiKu",$db);
//查詢用戶是否存在
$result=mysql_query("SELECT * FROM user where name='$PHP_AUTH_USER' and password='$PHP_AUTH_PW'",$db);
if ($myrow = mysql_fetch_row($result))
{
//以下為身份驗(yàn)證成功后的相關(guān)操作
...
}
else
{
//身份驗(yàn)證不成功,提示用戶重新輸入
header("WWW-Authenticate:Basic realm="身份驗(yàn)證功能"");
header("HTTP/1.0 401 Unauthorized");
echo "身份驗(yàn)證失敗,您無(wú)權(quán)共享網(wǎng)絡(luò)資源!";
exit();
}
?>
程序說(shuō)明:
在程序中,首先檢查變量$PHP_AUTH_USER是否已經(jīng)設(shè)置。如果沒(méi)有設(shè)置,說(shuō)明需要驗(yàn)證,腳本發(fā)出HTTP 401錯(cuò)誤號(hào)頭標(biāo),告訴客戶端的瀏覽器需要進(jìn)行身份驗(yàn)證,由客戶端的瀏覽器彈出一個(gè)身份驗(yàn)證窗口,提示用戶輸入用戶名和密碼,輸入完成后,連接數(shù)據(jù)庫(kù),查詢?cè)撚糜脩裘懊艽a是否正確,如果正確,允許登錄進(jìn)行相關(guān)操作,如果不正確,繼續(xù)要求用戶輸入用戶名和密碼。
函數(shù)說(shuō)明:
1、isset():用于確定某個(gè)變量是否已被賦值。根據(jù)變量值是否存在,返回true或false
2、header():用于發(fā)送特定的HTTP標(biāo)頭。注意,使用header()函數(shù)時(shí),一定要在任何產(chǎn)生實(shí)際輸出的HTML或PHP代碼前面調(diào)用該函數(shù)。
3、mysql_connect():打開(kāi) MySQL 服務(wù)器連接。
4、mysql_db_query():送查詢字符串 (query) 到 MySQL 數(shù)據(jù)庫(kù)。
5、mysql_fetch_row():返回單列的各字段。
二、用session實(shí)現(xiàn)服務(wù)器驗(yàn)證
對(duì)于需要身份驗(yàn)證的頁(yè)面,使用apache服務(wù)器驗(yàn)證是最好不過(guò)的了。但是,apache服務(wù)器驗(yàn)證的界面不夠友好。而且,cgi模式的php,iis下的php,都不能使用apache服務(wù)器驗(yàn)證。這樣,我們可以利用session在不同頁(yè)面間保存用戶身份,達(dá)到身份驗(yàn)證的目的。
在后端我們同樣利用上面的Mysql數(shù)據(jù)庫(kù)存放用戶信息。
我們先編寫(xiě)一個(gè)用戶登錄界面,文件名為login.php,代碼職下:
<form action="login1.php">
用戶名:<input type="text" name="name"><br>
口 令:<input type="text" name="pass"><br>
<input type="submit" value="登錄">
</form>
login1.php處理提交的表單,代碼如下:
$db=mysql_connect("localhost","root","");
mysql_select_db("XinXiKu",$db);
$result=mysql_query("SELECT * FROM user where name='$name' and password='$pass'",$db);
if ($myrow = mysql_fetch_row($result))
{
//注冊(cè)用戶
session_start();
session_register("user");
$user=$myrow["user"];
// 身份驗(yàn)證成功,進(jìn)行相關(guān)操作
...
}
else
{
echo"身份驗(yàn)證失敗,您無(wú)權(quán)共享網(wǎng)絡(luò)資源!";
}
?>
這里需要說(shuō)明的是,用戶可以使用在后續(xù)的操作中用**http://domainname/next.php?user=用戶名 **來(lái)繞過(guò)身份驗(yàn)證。所以,后續(xù)的操作應(yīng)先檢查變量是否注冊(cè):已注冊(cè),則進(jìn)行相應(yīng)操作,否則視為非法登錄。相關(guān)代碼如下:
session_start();
if (!session_is_registered("user"))
{
echo "身份驗(yàn)證失敗,屬于非法登錄!";
}
else
{
//成功登錄進(jìn)行相關(guān)操作
...
}
?>
附錄:PHP以模塊方式安裝方法
1、首先下載文件:mod_php4-4.0.1-pl2。[如果你的不是PHP4,那么就趕快升級(jí)吧!]
解開(kāi)后有三個(gè)文件:mod_php4.dll、mod_php4.conf、readme.txt
2、相關(guān)文件拷貝
把mod_php4.dll拷貝到apache安裝目錄的modules目錄下面
把mod_php4.conf拷貝到apache安裝目錄的conf目錄下面
把msvcrt.dll文件拷貝到apache的安裝目錄下面
3、打開(kāi)conf/srm.conf文件 ,在其中加上一句
Include conf/mod_php4.conf
在做這一些之前請(qǐng)把您的httpd.conf中關(guān)于CGI模式的所以設(shè)置語(yǔ)句都去掉,即類(lèi)似下面的部分!
ScripAlias /php4/ "C:/php4/"
AddType application/x-httpd-php4 .php
AddType application/x-httpd-php4 .php3
AddType application/x-httpd-php4 .php4
Action application/x-httpd-php4 /php4/php.exe
要想使PHP支持更多的后綴名,沒(méi)問(wèn)題。在給出的配置文件mod_php4.conf已經(jīng)支持了三種后綴名php,php3,php4,如果你還想支持更多的后綴名可以更改這個(gè)文件,很簡(jiǎn)單的。
4、測(cè)試
用<? phpinfo(); ?> 測(cè)試。會(huì)看到Server API的值為apache,而不是cgi ,而且還有有關(guān)HTTP Headers Information的信息。
一、用HTTP標(biāo)頭來(lái)實(shí)現(xiàn)
標(biāo)頭是服務(wù)器以HTTP協(xié)議傳送HTML信息到瀏覽器前所送出的字串。HTTP采用一種挑戰(zhàn)/響應(yīng)模式對(duì)試圖進(jìn)入受密碼保護(hù)區(qū)域的用戶進(jìn)行身份驗(yàn)證。具體來(lái)說(shuō),當(dāng)用戶首次向WEB服務(wù)器發(fā)出訪問(wèn)受保護(hù)區(qū)域的請(qǐng)求時(shí),挑戰(zhàn)進(jìn)程被啟動(dòng),服務(wù)器返回特殊的401標(biāo)頭,表明該用戶身份未經(jīng)驗(yàn)證??蛻舳藶g覽器在檢測(cè)到上述響應(yīng)之后自動(dòng)彈出對(duì)話框,要求用戶輸入用戶名和密碼。用戶完成輸入之后點(diǎn)擊確定,其身份識(shí)別信息就被傳送到服務(wù)端進(jìn)行驗(yàn)證。如果用戶輸入的用戶名和密碼有效,WEB服務(wù)器將允許用戶進(jìn)入受保護(hù)區(qū)域,并且在整個(gè)訪問(wèn)過(guò)程中保持其身份的有效性。相反,若用戶輸入的用戶名稱(chēng)或密碼無(wú)法通過(guò)驗(yàn)證,客戶端瀏覽器會(huì)不斷彈出輸入窗口要求用戶再次嘗試輸入正確的信息。整個(gè)過(guò)程將一直持續(xù)到用戶輸入正確的信息位置,也可以設(shè)定允許用戶進(jìn)行嘗試的最大次數(shù),超出時(shí)將自動(dòng)拒絕用戶的訪問(wèn)請(qǐng)求。
在PHP腳本中,使用函數(shù)header()直接給客戶端的瀏覽器發(fā)送HTTP標(biāo)頭,這樣在客戶端將會(huì)自動(dòng)彈出用戶名和密碼輸入窗口,來(lái)實(shí)現(xiàn)我們的身份認(rèn)證功能。在PHP中,客戶端用戶輸入的信息傳送到服務(wù)器之后自動(dòng)保存在 $PHP_AUTH_USER,$PHP_AUTH_PW,以及 $PHP_AUTH_TYPE這三個(gè)全局變量中。利用這三個(gè)變量,我們可以根據(jù)保存在數(shù)據(jù)文件或者數(shù)據(jù)庫(kù)中用戶帳號(hào)信息來(lái)驗(yàn)證用戶身份!
不過(guò),需要提醒使用者注意的是:只有在以模塊方式安裝的PHP中才能使用$PHP_AUTH_USER,$PHP_AUTH_PW,以及 $PHP_AUTH_TYPE這三個(gè)變量。如果用戶使用的是CGI模式的PHP則無(wú)法實(shí)現(xiàn)驗(yàn)證功能。在本節(jié)后附有PHP的模塊方式安裝方法。
下面我們用Mysql數(shù)據(jù)庫(kù)來(lái)存儲(chǔ)用戶的身份。我們需要從數(shù)據(jù)庫(kù)中提取每個(gè)帳號(hào)的用戶名和密碼以便與$PHP_AUTH_USER和$PHP_AUTH_PW變量進(jìn)行比較,判斷用戶的真實(shí)性。
首先,在MySql中建立一個(gè)存放用戶信息的數(shù)據(jù)庫(kù)
數(shù)據(jù)庫(kù)名為XinXiKu ,表名為user;表定義如下:
復(fù)制代碼 代碼如下:
create table user(
ID INT(4) NOT NULL AUTO_INCREMENT,
name VARCHAR(8) NOT NULL,
password CHAR(8) NOT NULL,
PRIMARY KEY(ID)
)
說(shuō)明:
1、ID為一個(gè)序列號(hào),不為零而且自動(dòng)遞增,為主鍵;
2、name為用戶名,不能為空;
3、password為用戶密碼,不能為空;
以下是用戶驗(yàn)證文件login.php
復(fù)制代碼 代碼如下:
//判斷用戶名是否設(shè)置
if(!isset($PHP_AUTH_USER))
{
header("WWW-Authenticate:Basic realm="身份驗(yàn)證功能"");
header("HTTP/1.0 401 Unauthorized");
echo "身份驗(yàn)證失敗,您無(wú)權(quán)共享網(wǎng)絡(luò)資源!";
exit();
}
/*連接數(shù)據(jù)庫(kù)*/
$db=mysql_connect("localhost","root","");
//選擇數(shù)據(jù)庫(kù)
mysql_select_db("XinXiKu",$db);
//查詢用戶是否存在
$result=mysql_query("SELECT * FROM user where name='$PHP_AUTH_USER' and password='$PHP_AUTH_PW'",$db);
if ($myrow = mysql_fetch_row($result))
{
//以下為身份驗(yàn)證成功后的相關(guān)操作
...
}
else
{
//身份驗(yàn)證不成功,提示用戶重新輸入
header("WWW-Authenticate:Basic realm="身份驗(yàn)證功能"");
header("HTTP/1.0 401 Unauthorized");
echo "身份驗(yàn)證失敗,您無(wú)權(quán)共享網(wǎng)絡(luò)資源!";
exit();
}
?>
程序說(shuō)明:
在程序中,首先檢查變量$PHP_AUTH_USER是否已經(jīng)設(shè)置。如果沒(méi)有設(shè)置,說(shuō)明需要驗(yàn)證,腳本發(fā)出HTTP 401錯(cuò)誤號(hào)頭標(biāo),告訴客戶端的瀏覽器需要進(jìn)行身份驗(yàn)證,由客戶端的瀏覽器彈出一個(gè)身份驗(yàn)證窗口,提示用戶輸入用戶名和密碼,輸入完成后,連接數(shù)據(jù)庫(kù),查詢?cè)撚糜脩裘懊艽a是否正確,如果正確,允許登錄進(jìn)行相關(guān)操作,如果不正確,繼續(xù)要求用戶輸入用戶名和密碼。
函數(shù)說(shuō)明:
1、isset():用于確定某個(gè)變量是否已被賦值。根據(jù)變量值是否存在,返回true或false
2、header():用于發(fā)送特定的HTTP標(biāo)頭。注意,使用header()函數(shù)時(shí),一定要在任何產(chǎn)生實(shí)際輸出的HTML或PHP代碼前面調(diào)用該函數(shù)。
3、mysql_connect():打開(kāi) MySQL 服務(wù)器連接。
4、mysql_db_query():送查詢字符串 (query) 到 MySQL 數(shù)據(jù)庫(kù)。
5、mysql_fetch_row():返回單列的各字段。
二、用session實(shí)現(xiàn)服務(wù)器驗(yàn)證
對(duì)于需要身份驗(yàn)證的頁(yè)面,使用apache服務(wù)器驗(yàn)證是最好不過(guò)的了。但是,apache服務(wù)器驗(yàn)證的界面不夠友好。而且,cgi模式的php,iis下的php,都不能使用apache服務(wù)器驗(yàn)證。這樣,我們可以利用session在不同頁(yè)面間保存用戶身份,達(dá)到身份驗(yàn)證的目的。
在后端我們同樣利用上面的Mysql數(shù)據(jù)庫(kù)存放用戶信息。
我們先編寫(xiě)一個(gè)用戶登錄界面,文件名為login.php,代碼職下:
復(fù)制代碼 代碼如下:
<form action="login1.php">
用戶名:<input type="text" name="name"><br>
口 令:<input type="text" name="pass"><br>
<input type="submit" value="登錄">
</form>
login1.php處理提交的表單,代碼如下:
復(fù)制代碼 代碼如下:
$db=mysql_connect("localhost","root","");
mysql_select_db("XinXiKu",$db);
$result=mysql_query("SELECT * FROM user where name='$name' and password='$pass'",$db);
if ($myrow = mysql_fetch_row($result))
{
//注冊(cè)用戶
session_start();
session_register("user");
$user=$myrow["user"];
// 身份驗(yàn)證成功,進(jìn)行相關(guān)操作
...
}
else
{
echo"身份驗(yàn)證失敗,您無(wú)權(quán)共享網(wǎng)絡(luò)資源!";
}
?>
這里需要說(shuō)明的是,用戶可以使用在后續(xù)的操作中用**http://domainname/next.php?user=用戶名 **來(lái)繞過(guò)身份驗(yàn)證。所以,后續(xù)的操作應(yīng)先檢查變量是否注冊(cè):已注冊(cè),則進(jìn)行相應(yīng)操作,否則視為非法登錄。相關(guān)代碼如下:
復(fù)制代碼 代碼如下:
session_start();
if (!session_is_registered("user"))
{
echo "身份驗(yàn)證失敗,屬于非法登錄!";
}
else
{
//成功登錄進(jìn)行相關(guān)操作
...
}
?>
附錄:PHP以模塊方式安裝方法
1、首先下載文件:mod_php4-4.0.1-pl2。[如果你的不是PHP4,那么就趕快升級(jí)吧!]
解開(kāi)后有三個(gè)文件:mod_php4.dll、mod_php4.conf、readme.txt
2、相關(guān)文件拷貝
把mod_php4.dll拷貝到apache安裝目錄的modules目錄下面
把mod_php4.conf拷貝到apache安裝目錄的conf目錄下面
把msvcrt.dll文件拷貝到apache的安裝目錄下面
3、打開(kāi)conf/srm.conf文件 ,在其中加上一句
Include conf/mod_php4.conf
在做這一些之前請(qǐng)把您的httpd.conf中關(guān)于CGI模式的所以設(shè)置語(yǔ)句都去掉,即類(lèi)似下面的部分!
ScripAlias /php4/ "C:/php4/"
AddType application/x-httpd-php4 .php
AddType application/x-httpd-php4 .php3
AddType application/x-httpd-php4 .php4
Action application/x-httpd-php4 /php4/php.exe
要想使PHP支持更多的后綴名,沒(méi)問(wèn)題。在給出的配置文件mod_php4.conf已經(jīng)支持了三種后綴名php,php3,php4,如果你還想支持更多的后綴名可以更改這個(gè)文件,很簡(jiǎn)單的。
4、測(cè)試
用<? phpinfo(); ?> 測(cè)試。會(huì)看到Server API的值為apache,而不是cgi ,而且還有有關(guān)HTTP Headers Information的信息。
您可能感興趣的文章:
- 通過(guò)PHP修改Linux或Unix口令的方法分享
- ThinkPHP權(quán)限認(rèn)證Auth實(shí)例詳解
- 用PHP寫(xiě)的MySQL數(shù)據(jù)庫(kù)用戶認(rèn)證系統(tǒng)代碼
- PHP實(shí)現(xiàn)用戶認(rèn)證及管理完全源碼
- PHP防注入安全代碼
- 理解php Hash函數(shù),增強(qiáng)密碼安全
- php.ini 啟用disable_functions提高安全
- php中安全模式safe_mode配置教程
- PHP開(kāi)發(fā)不能違背的安全規(guī)則 過(guò)濾用戶輸入
- php集成動(dòng)態(tài)口令認(rèn)證
相關(guān)文章
php 定義404頁(yè)面的實(shí)現(xiàn)代碼
今天需要用php輸出一個(gè)404錯(cuò)誤頁(yè)面,然后加載404頁(yè)面輸出,各個(gè)方便給用戶與搜索引擎好的用戶體驗(yàn)2012-11-11
php utf-8轉(zhuǎn)unicode的函數(shù)
php下我們想把uft-8,轉(zhuǎn)成unicode可以用下面的函數(shù)來(lái)實(shí)現(xiàn)2008-06-06
PHP封裝的非對(duì)稱(chēng)加密RSA算法示例
這篇文章主要介紹了PHP封裝的非對(duì)稱(chēng)加密RSA算法,結(jié)合實(shí)例形式分析了php實(shí)現(xiàn)的RSA加密算法類(lèi)及其相關(guān)使用技巧,需要的朋友可以參考下2018-05-05
PHP XML Expat解析器知識(shí)點(diǎn)總結(jié)
在本文里我們給大家整理了關(guān)于PHP XML Expat解析器的相關(guān)知識(shí)點(diǎn)內(nèi)容,需要的朋友們學(xué)習(xí)下這個(gè)吧。2019-02-02
php實(shí)現(xiàn)隨機(jī)顯示圖片方法匯總
本文分享一個(gè)php實(shí)現(xiàn)的隨機(jī)顯示圖片的函數(shù),可以將指定文件夾中存放的圖片隨機(jī)地顯示出來(lái)。有興趣的朋友研究下吧。2015-05-05

