| 
 UID2033152 威望1 点 金钱3090 金币 点卡0 点 
 | 
1#
 发表于 2004-11-17 18:00 
 | 只看该作者 
 [转帖]验证码技术及其实现
| 目前,不少网站为了防止用户利用机器人自动注册、登录、灌水,都采用了验证码技术。所谓验证码,就是将一串随机产生的数字或符号,生成一幅图片,图片里加上一些干扰象素(防止OCR),由用户肉眼识别其中的验证码信息,输入表单提交网站验证,验证成功后才能使用某项功能。 PHP实现:
 我们这里展示了如何编写PHP程序实现验证码功能:
 
 代码一:
 本文程序在Apache 2.0.45 + PHP 4.3.1环境下运行通过。复制代码          <?php           /*           *  Filename:  authpage.php           *  Author:  hutuworm           *  Date:  2003-04-28           *  @Copyleft  hutuworm.org           */              srand((double)microtime()*1000000);              //验证用户输入是否和验证码一致             if(isset($HTTP_POST_VARS[';authinput';]))              {                 if(strcmp($HTTP_POST_VARS[';authnum';],$HTTP_POST_VARS[';authinput';])==0)                        echo "验证成功!";                 else                     echo "验证失败!";             }                      //生成新的四位整数验证码             while(($authnum=rand()%10000)<1000);            ?>             <form action=authpage.php method=post>             <table>                 请输入验证码:<input type=text name=authinput style="width:         80px"><br>                 <input type=submit name="验证" value="提交验证码">                 <input type=hidden name=authnum value=<? echo $authnum; ?>>                 <img src=authimg.php?authnum=<? echo $authnum; ?>>             </table>             </form>            代码二:            <?php           /*           *  Filename:  authimg.php           *  Author:  hutuworm           *  Date:  2003-04-28           *  @Copyleft  hutuworm.org           */              //生成验证码图片             Header("Content-type: image/PNG");              srand((double)microtime()*1000000);             $im = imagecreate(58,28);             $black = ImageColorAllocate($im, 0,0,0);             $white = ImageColorAllocate($im, 255,255,255);             $gray = ImageColorAllocate($im, 200,200,200);             imagefill($im,68,30,$gray);              //将四位整数验证码绘入图片             imagestring($im, 5, 10, 8, $HTTP_GET_VARS[';authnum';], $black);                for($i=0;$i<50;$i++)  //加入干扰象素             {                 imagesetpixel($im, rand()%70 , rand()%30 , $black);             }                ImagePNG($im);             ImageDestroy($im);         ?>
 上文只是对验证码功能的一个简单实现,并没有考虑商用安全性问题。如果要增强安全性,将此功能投入商业应用,则可以通过以下几个步骤实现:
 
 1. 启用Session。
 2. authnum在authimg.php中生成,并计算md5sum,存入session。
 3.
 authpage.php将authinput计算md5sum后,与session中的authnum(md5sum)对比得出验证结果。
 
 
 本站注:作者使用了简单的代码实现了很酷的功能。不过在添加干扰像素时的效果不是太好,大家可以看一下雨声论坛登录时的效验码(http://ror.cn/perl/ut/user_login.cgi),偶把第二段代码稍改了一下,生成了与其类似的效果。
 
 修改后的代码如下:
 显示结果如下图:复制代码        <?php         /*         *  Filename: authimg.php         *  Author:  hutuworm         *  Date:   2003-04-28         *  @Copyleft hutuworm.org         */         //生成验证码图片         Header("Content-type: image/PNG");          srand((double)microtime()*1000000);         $im = imagecreate(62,20);         $black = ImageColorAllocate($im, 0,0,0);         $white = ImageColorAllocate($im, 255,255,255);         $gray = ImageColorAllocate($im, 200,200,200);         imagefill($im,68,30,$gray);         while(($authnum=rand()%100000)<10000);         //将四位整数验证码绘入图片         imagestring($im, 5, 10, 3, $authnum, $black);         for($i=0;$i<200;$i++)  //加入干扰象素         {           $randcolor =         ImageColorallocate($im,rand(0,255),rand(0,255),rand(0,255));           imagesetpixel($im, rand()%70 , rand()%30 , $randcolor);         }         ImagePNG($im);         ImageDestroy($im);         ?> 
 有兴趣的朋友可以自己试一下。
 看一下上面这段代码:
 第一次看PHP代码,呵呵。奇怪的学习经历。
 代码一:
 代码二:复制代码            <?php             /*             *  Filename:  authpage.php             *  Author:  hutuworm             *  Date:  2003-04-28             *  @Copyleft  hutuworm.org             */      //以系统时间为种子得到随机数   //srand -- 播下随机数发生器种子   //microtime -- 返回当前 UNIX 时间戳和微秒数                srand((double)microtime()*1000000);                  //验证用户输入是否和验证码一致               if(isset($HTTP_POST_VARS[';authinput';]))                {   //authnum的值是不应该发送到客户端的,这里应该是为了构建一个简单的例子先吧   //authnum 保存在 session 里应该是个可行的办法                      if(strcmp($HTTP_POST_VARS[';authnum';],$HTTP_POST_VARS[';authinput';])==0)                            echo "验证成功!";                   else                       echo "验证失败!";               }                          //生成新的四位整数验证码               while(($authnum=rand()%10000)<1000);              ?>               <form action=authpage.php method=post>               <table>                   请输入验证码:<input type=text name=authinput style="width:           80px"><br>                   <input type=submit name="验证" value="提交验证码">                   <input type=hidden name=authnum value=<? echo $authnum; ?>>                   <img src=authimg.php?authnum=<? echo $authnum; ?>>               </table>               </form> 
本文程序在Apache 2.0.45 + PHP 4.3.1环境下运行通过。复制代码          <?php             /*             *  Filename:  authimg.php             *  Author:  hutuworm             *  Date:  2003-04-28             *  @Copyleft  hutuworm.org             */      //下面几行代码肯定会让写ASP的GGDD们羡慕不已。                  //生成验证码图片      //指定头信息                    Header("Content-type: image/PNG");                srand((double)microtime()*1000000);      //返回一个图像标识符,代表了一幅大小为58 X 28的空白图像。                 $im = imagecreate(58,28);               $black = ImageColorAllocate($im, 0,0,0);               $white = ImageColorAllocate($im, 255,255,255);               $gray = ImageColorAllocate($im, 200,200,200);               imagefill($im,68,30,$gray);         //强悍!!                    //将四位整数验证码绘入图片      //方便起见,使可以通过authimg.php?authnum=XXXX的方式得到图片。实际应用中这种做法肯定是不足取的。                   imagestring($im, 5, 10, 8, $HTTP_GET_VARS[';authnum';], $black);                    for($i=0;$i<50;$i++)  //加入干扰象素               {                   imagesetpixel($im, rand()%70 , rand()%30 , $black);               }                    ImagePNG($im);   //imagepng -- 以 PNG 格式将图像输出到浏览器或文件               ImageDestroy($im);           ?> 
 上文只是对验证码功能的一个简单实现,并没有考虑商用安全性问题。如果要增强安全性,将此功能投入商业应用,则可以通过以下几个步骤实现:
 
 1. 启用Session。
 2. authnum在authimg.php中生成,并计算md5sum,存入session。
 3. authpage.php将authinput计算md5sum后,与session中的authnum(md5sum)对比得出验证结果。
 
 
 本站注:作者使用了简单的代码实现了很酷的功能。不过在添加干扰像素时的效果不是太好,大家可以看一下雨声论坛登录时的效验码(http://ror.cn/perl/ut/user_login.cgi),偶把第二段代码稍改了一下,生成了与其类似的效果。
 
 修改后的代码如下:
 验证码技术核心内容是动态生成图片。复制代码          <?php           /*           *  Filename: authimg.php           *  Author:  hutuworm           *  Date:   2003-04-28           *  @Copyleft hutuworm.org           */           //生成验证码图片           Header("Content-type: image/PNG");            srand((double)microtime()*1000000);           $im = imagecreate(62,20);           $black = ImageColorAllocate($im, 0,0,0);           $white = ImageColorAllocate($im, 255,255,255);           $gray = ImageColorAllocate($im, 200,200,200);           imagefill($im,68,30,$gray);           while(($authnum=rand()%100000)<10000);           //将四位整数验证码绘入图片           imagestring($im, 5, 10, 3, $authnum, $black);           for($i=0;$i<200;$i++)  //加入干扰象素           {             $randcolor =           ImageColorallocate($im,rand(0,255),rand(0,255),rand(0,255));             imagesetpixel($im, rand()%70 , rand()%30 , $randcolor);           }           ImagePNG($im);           ImageDestroy($im);           ?> 
 | 
 |