php基于表单密码验证与HTTP验证用法实例


Posted in PHP onJanuary 06, 2015

本文实例讲述了php基于表单密码验证与HTTP验证用法。分享给大家供大家参考。具体分析如下:

PHP 的 HTTP 认证机制仅在 PHP 以 Apache 模块方式运行时才有效,因此该功能不适用于 CGI 版本。在 Apache 模块的 PHP 脚本中,可以用 header() 函数来向客户端浏览器发送“Authentication Required”信息,使其弹出一个用户名/密码输入窗口。当用户输入用户名和密码后,包含有 URL 的 PHP 脚本将会加上预定义变量 PHP_AUTH_USER , PHP_AUTH_PW 和 AUTH_TYPE 被再次调用,这三个变量分别被设定为用户名,密码和认证类型。预定义变量保存在 $_SERVER 或者 $HTTP_SERVER_VARS 数组中。支持“Basic”和“Digest”(自 PHP 5.1.0 起)认证方法。感兴趣的朋友可以参阅header()函数相关信息。

PHP 版本问题: Autoglobals 全局变量,包括 $_SERVER 等,自 PHP 4.1.0 起有效, $HTTP_SERVER_VARS 从 PHP 3 开始有效。

以下是在页面上强迫客户端认证的脚本范例.

例子 34-1. Basic HTTP 认证范例

<?php 

   if (!isset( $_SERVER [ 'PHP_AUTH_USER' ])) {  

     header ( 'WWW-Authenticate: Basic realm="My Realm"' );  

     header ( 'HTTP/1.0 401 Unauthorized' );  

    echo 'Text to send if user hits Cancel button' ;  

    exit; 

  } else {  

    echo "<p>Hello { $_SERVER [ 'PHP_AUTH_USER' ]} .</p>" ;  

    echo "<p>You entered { $_SERVER [ 'PHP_AUTH_PW' ]} as your password.</p>" ;  

  }  

?>

例子 34-2. Digest HTTP 认证范例

本例演示怎样实现一个简单的 Digest HTTP 认证脚本,更多信息请参考 RFC 2617.

<?php  

$realm = 'Restricted area' ; 

//user => password  

$users = array( 'admin' => 'mypass' , 'guest' => 'guest' ); 

 

if (!isset( $_SERVER [ 'PHP_AUTH_DIGEST' ])) {  

     header ( 'HTTP/1.1 401 Unauthorized' );  

     header ( 'WWW-Authenticate: Digest realm="' . $realm .  

            '" qop="auth" nonce="' . uniqid (). '" opaque="' . md5 ( $realm ). '"' ); 

    die( 'Text to send if user hits Cancel button' );  

} 

// analize the PHP_AUTH_DIGEST variable  

preg_match ( '/username="(?P<username>.*)",s*realm="(?P<realm>.*)",s*nonce="(?P<nonce>.*)",s*uri="(?P<uri>.*)",s*response="(?P<response>.*)",s*opaque="(?P<opaque>.*)",s*qop=(?P<qop>.*),s*nc=(?P<nc>.*),s*cnonce="(?P<cnonce>.*)"/' , $_SERVER [ 'PHP_AUTH_DIGEST' ], $digest ); 

if (!isset( $users [ $digest [ 'username' ]]))  

    die( 'Username not valid!' ); 

 

// generate the valid response  

$A1 = md5 ( $digest [ 'username' ] . ':' . $realm . ':' . $users [ $digest [ 'username' ]]);  

$A2 = md5 ( $_SERVER [ 'REQUEST_METHOD' ]. ':' . $digest [ 'uri' ]);  

$valid_response = md5 ( $A1 . ':' . $digest [ 'nonce' ]. ':' . $digest [ 'nc' ]. ':' . $digest [ 'cnonce' ]. ':' . $digest [ 'qop' ]. ':' . $A2 ); 

if ( $digest [ 'response' ] != $valid_response )  

    die( 'Wrong Credentials!' ); 

// ok, valid username & password  

echo 'Your are logged in as: ' . $digest [ 'username' ]; 

?>

兼容性问题:在编写 HTTP 标头代码时请格外小心,为了对所有的客户端保证兼容性,关键字“Basic”的第一个字母必须大写为“B”,分界字符串必须用双引号(不是单引号)引用;并且在标头行 HTTP/1.0 401 中,在 401 前必须有且仅有一个空格.

在以上例子中,仅仅只打印出了 PHP_AUTH_USER 和 PHP_AUTH_PW 的值,但在实际运用中,可能需要对用户名和密码的合法性进行检查,或许进行数据库教程的查询,或许从 dbm 文件中检索.

注意有些 Internet Explorer 浏览器本身有问题。它对标头的顺序显得似乎有点吹毛求疵。目前看来在发送 HTTP/1.0 401 之前先发送 WWW-Authenticate 标头似乎可以解决此问题。

自 PHP 4.3.0 起,为了防止有人通过编写脚本来从用传统外部机制认证的页面上获取密码,当外部认证对特定页面有效,并且 安全模式 被开启时,PHP_AUTH 变量将不会被设置,但无论如何, REMOTE_USER 可以被用来辨认外部认证的用户,因此可以用 $_SERVER['REMOTE_USER'] 变量.

配置说明:PHP 用是否有 AuthType 指令来判断外部认证机制是否有效。

注意,这仍然不能防止有人通过未认证的 URL 来从同一服务器上认证的 URL 上偷取密码.

Netscape Navigator 和 Internet Explorer 浏览器都会在收到 401 的服务端返回信息时清空所有的本地浏览器整个域的 Windows 认证缓存,这能够有效的注销一个用户,并迫使他们重新输入他们的用户名和密码,有些人用这种方法来使登录状态“过期”,或者作为“注销”按钮的响应行为.

例子 34-3.强迫重新输入用户名和密码的 HTTP 认证的范例

<?php  

   function authenticate () {  

     header ( 'WWW-Authenticate: Basic realm="Test Authentication System"' ); 

     header ( 'HTTP/1.0 401 Unauthorized' );  

    echo "You must enter a valid login ID and password to access this resourcen" ;  

    exit;  

  } 

  if (!isset( $_SERVER [ 'PHP_AUTH_USER' ]) ||  

      ( $_POST [ 'SeenBefore' ] == 1 && $_POST [ 'OldAuth' ] == $_SERVER [ 'PHP_AUTH_USER' ])) {  

    authenticate ();  

  }  

  else {  

   echo "<p>Welcome: { $_SERVER [ 'PHP_AUTH_USER' ]} <br />" ;  

   echo "Old: { $_REQUEST [ 'OldAuth' ]} " ;  

   echo "<form action=' { $_SERVER [ 'PHP_SELF' ]} ' METHOD='post'> n " ;  

   echo "<input type='hidden' name='SeenBefore' value='1' />n" ;  

   echo "<input type='hidden' name='OldAuth' value=' { $_SERVER [ 'PHP_AUTH_USER' ]} ' /> n " ;  

   echo "<input type='submit' value='Re Authenticate' />n" ;  

   echo "</form></p>n" ;  

  } 

?>

该行为对于 HTTP 的 Basic 认证标准来说并不是必须的,因此不能依靠这种方法,对 Lynx 浏览器的测试表明 Lynx 在收到 401 的服务端返回信息时不会清空认证文件,因此只要对认证文件的检查要求没有变化,只要用户点击“后退”按钮,再点击“前进”按钮,其原有资源仍然能够被访问,不过,用户可以通过按“_”键来清空他们的认证信息.

在下例中,我们是使用$PHP_AUTH_USER和$PHP_AUTH_PW这两个变量来验证进入者是否合法并允许进入。在本例中被允许登录的用户名称和密码对分别为tnc和nature:

<?php 

if(!isset($PHP_AUTH_USER))  

{  

Header("WWW-Authenticate: Basic realm="My Realm"");  

Header("HTTP/1.0 401 Unauthorized");  

echo "Text to send if user hits Cancel buttonn";  

exit;  

}  

else  

{  

if ( !($PHP_AUTH_USER=="tnc" && $PHP_AUTH_PW=="nature") )  

{  

// 如果是错误的用户名称/密码对,强制再验证  

Header("WWW-Authenticate: Basic realm="My Realm"");  

Header("HTTP/1.0 401 Unauthorized");  

echo "ERROR : $PHP_AUTH_USER/$PHP_AUTH_PW is invalid.";  

exit;  

}  

else  

{  

echo "Welcome tnc!";  

}  

?>

事实上再实际引用中不大可能如上面使用代码段明显的用户名称/密码对,而是利用数据库或者加密的密码文件存取它们.

根据指定的验证信息核实用户身份:

首先,我们可以使用以下代码确定用户是否已经输入了用户名和密码,并显示出用户输入的信息.

<?php  

if (!isset($PHP_AUTH_USER)) {  

header('WWW-Authenticate: Basic realm="My Private Stuff"');  

header('HTTP/1.0 401 Unauthorized');  

echo 'Authorization Required.';  

exit;  

}  

else {  

echo "<P>You have entered this username: $PHP_AUTH_USER<br>  

You have entered this password: $PHP_AUTH_PW<br>  

The authorization type is: $PHP_AUTH_TYPE</p>";  

}  

?>

说明:

isset()函数用于确定某个变量是否已被赋值,根据变量值是否存在,返回true或false.

header()函数用于发送特定的HTTP标头,注意,使用header()函数时,一定要在任何产生实际输出的HTML或PHP代码前面调用该函数.

虽然上述代码相当简单,没有根据任何实际值对用户输入的用户名和密码进行有效验证,但是至少我们了解了如何使用PHP在客户端产生输入对话框.

下面,我们就来了解一下如何根据指定的验证信息核实用户身份,代码如下:

<?php  

if (!isset($PHP_AUTH_USER)) {  

header('WWW-Authenticate: Basic realm="My Private Stuff"');  

header('HTTP/1.0 401 Unauthorized');  

echo 'Authorization Required.';  

exit;  

}  

else if (isset($PHP_AUTH_USER)) {  

if (($PHP_AUTH_USER != "admin") || ($PHP_AUTH_PW != "123")) {  

header('WWW-Authenticate: Basic realm="My Private Stuff"');  

header('HTTP/1.0 401 Unauthorized');  

echo 'Authorization Required.';  

exit;  

} else {  

echo "<P>You're authorized!</p>";  

}  

}  

?>

在这里,我们首先检查用户是否已经输入了用户名称和密码,如果没有则弹出相应对话框要求用户输入身份信息,随后,我们通过判断用户输入的信息是否符合admin/123这一指定用户帐号来授予用户访问权限或提示用户再次输入正确的信息,这种方法适用于所有用户都使用同一登录帐号的网站.

另一种简易的密码验证

如果你是在windows98下面编写和运行着你的PHP脚本,或者是你在Linux下面按默认设置,将PHP安装成一个CGI程序的话,你将无法使用上面的PHP程序来实现验证功能,为此,无边给大家提供了另外一种简易的密码验证的方法,虽然实用性不大,但是拿来学习还是挺好的.

<?php 

if($_POST[Submit]=="提交"){  //如果用户提交了数据,则执行操作 

$password=$_POST[password];
//获取用户输入的数据,并保存在变量 password 中 

$cpassword=$_POST[cpassword];   //获取用户输入的确认数据,保存在变量 $cpassord 中 

if(emptyempty($password) || emptyempty($cpassword)) 

{ 

    die("密码不可空!"); 

} 

elseif ( ((strlen($password) < 5) || (strlen($password) > 15))) 

{ 

    die("密码长度在5和15之间"); 

} 

   //--- 值比较 

elseif ( !(strlen($password) == strlen($cpassword)) ) 

{ 

    die("两次输入密码不匹配! "); 

} 

elseif( !($password === $cpassword)) //值和数据类型比较 

{ 

   die("两次密码不匹配! "); 

} 

else  //循环输出密码,因为是密码所以输出*号  

{ 

    for ($i=0;$i<strlen($password);$i++) 

      { 

            echo "*"; 

      } 

} 

} 

?> 

<html> 

<head> 

<meta http-equiv="Content-Type" content="text/html; charset=gb2312"> 

<title>表单验证-密码字段验证</title> 

</head> 

<body> 

<form name="form1" method="post" action="<?=$_SERVER['PHP_SELF'] ?>"> 

请输入密码:<input type="text" name="password"><br> 

确认密码:<input type="password" name="cpassword"><br> 

<input type="submit" name="Submit" value="提交"> 

</form> 

</body> 

</html>

希望本文所述对大家的php程序设计有所帮助。

PHP 相关文章推荐
PHP4实际应用经验篇(8)
Oct 09 PHP
thinkphp3.0 模板中函数的使用
Nov 13 PHP
PHP的引用详解
Feb 22 PHP
php获取从html表单传递数组的方法
Mar 20 PHP
php如何实现只替换一次或N次
Oct 29 PHP
PHP中使用array函数新建一个数组
Nov 19 PHP
php实现贪吃蛇小游戏
Jul 26 PHP
php实现连接access数据库并转txt写入的方法
Feb 08 PHP
Laravel利用gulp如何构建前端资源详解
Jun 03 PHP
PHP检查端口是否可以被绑定的方法示例
Aug 09 PHP
Yii框架数据库查询、增加、删除操作示例
Oct 14 PHP
YII2框架中actions的作用与使用方法示例
Mar 13 PHP
php使用fputcsv()函数csv文件读写数据的方法
Jan 06 #PHP
phplot生成图片类用法详解
Jan 06 #PHP
写一段简单的PHP建立文件夹代码
Jan 06 #PHP
php读取flash文件高宽帧数背景颜色的方法
Jan 06 #PHP
php自动获取关键字的方法
Jan 06 #PHP
windows7下php开发环境搭建图文教程
Jan 06 #PHP
PHP中$this和$that指针使用实例
Jan 06 #PHP
You might like
php写的AES加密解密类分享
2014/06/20 PHP
一个非常完美的读写ini格式的PHP配置类分享
2015/02/12 PHP
php session 写入数据库
2016/02/13 PHP
php递归函数怎么用才有效
2018/02/24 PHP
thinkPHP5框架实现基于ajax的分页功能示例
2018/06/12 PHP
数理公式,也可以这么唯美
2021/03/10 无线电
客户端限制只能上传jpg格式图片的js代码
2010/12/09 Javascript
Javascript处理DOM元素事件实现代码
2012/05/23 Javascript
JavaScript检查弹出窗口是否被阻拦的方法技巧
2015/03/13 Javascript
Javascript 计算字符串在localStorage中所占字节数
2015/10/21 Javascript
angularJS 如何读写缓冲的方法(推荐)
2016/08/06 Javascript
javascript比较语义化版本号的实现代码
2016/09/09 Javascript
JavaScript表单验证开发
2016/11/23 Javascript
jQuery文字轮播特效
2017/02/12 Javascript
ztree简介_动力节点Java学院整理
2017/07/19 Javascript
React进阶学习之组件的解耦之道
2017/08/07 Javascript
vue-cli2.0转3.0之项目搭建的详细步骤
2018/12/11 Javascript
基于vue-cli搭建多模块且各模块独立打包的项目
2019/06/12 Javascript
layui use 定义js外部引用函数的方法
2019/09/26 Javascript
jstree中的checkbox默认选中和隐藏示例代码
2019/12/29 Javascript
Openlayers3实现车辆轨迹回放功能
2020/09/29 Javascript
python中函数默认值使用注意点详解
2016/06/01 Python
关于Python 3中print函数的换行详解
2017/08/08 Python
魔幻般冒泡背景的CSS3按钮动画
2016/02/27 HTML / CSS
.net笔试题
2014/03/03 面试题
母亲节演讲稿范文
2014/01/02 职场文书
生物制药专业求职信
2014/03/11 职场文书
《宋庆龄故居的樟树》教学反思
2014/04/07 职场文书
服装设计师求职信
2014/06/04 职场文书
教育系统干部作风整顿心得体会
2014/09/09 职场文书
法院执行局工作总结
2015/08/11 职场文书
2016年学校党支部公开承诺书
2016/03/25 职场文书
《艾尔登法环》发布最新「战技」宣传片
2022/04/03 其他游戏
清空 Oracle 安装记录并重新安装
2022/04/26 Oracle
MySQL普通表如何转换成分区表
2022/05/30 MySQL
SpringBoot使用ip2region获取地理位置信息的方法
2022/06/21 Java/Android