返回列表 回复 发帖

[转帖]验证码技术及其实现

目前,不少网站为了防止用户利用机器人自动注册、登录、灌水,都采用了验证码技术。所谓验证码,就是将一串随机产生的数字或符号,生成一幅图片,图片里加上一些干扰象素(防止OCR),由用户肉眼识别其中的验证码信息,输入表单提交网站验证,验证成功后才能使用某项功能。
PHP实现:
  我们这里展示了如何编写PHP程序实现验证码功能:
   
        代码一: 
  1.           <?php
  2.           /*
  3.           *  Filename:  authpage.php
  4.           *  Author:  hutuworm
  5.           *  Date:  2003-04-28
  6.           *  @Copyleft  hutuworm.org
  7.           */
  8.    
  9.           srand((double)microtime()*1000000);
  10.    
  11.           //验证用户输入是否和验证码一致
  12.             if(isset(&#36;HTTP_POST_VARS[';authinput';])) 
  13.             {
  14.                 if(strcmp(&#36;HTTP_POST_VARS[';authnum';],&#36;HTTP_POST_VARS[';authinput';])==0)
  15.    
  16.                     echo "验证成功!";
  17.                 else
  18.                     echo "验证失败!";
  19.             }
  20.           
  21.           //生成新的四位整数验证码
  22.             while((&#36;authnum=rand()%10000)<1000); 
  23.           ?>
  24.             <form action=authpage.php method=post>
  25.             <table>
  26.                 请输入验证码:<input type=text name=authinput style="width:
  27.         80px"><br>
  28.                 <input type=submit name="验证" value="提交验证码">
  29.                 <input type=hidden name=authnum value=<? echo &#36;authnum; ?>>
  30.                 <img src=authimg.php?authnum=<? echo &#36;authnum; ?>>
  31.             </table>
  32.             </form>
  33.    
  34.         代码二:
  35.    
  36.         <?php
  37.           /*
  38.           *  Filename:  authimg.php
  39.           *  Author:  hutuworm
  40.           *  Date:  2003-04-28
  41.           *  @Copyleft  hutuworm.org
  42.           */
  43.    
  44.           //生成验证码图片
  45.             Header("Content-type: image/PNG"); 
  46.             srand((double)microtime()*1000000);
  47.             &#36;im = imagecreate(58,28);
  48.             &#36;black = ImageColorAllocate(&#36;im, 0,0,0);
  49.             &#36;white = ImageColorAllocate(&#36;im, 255,255,255);
  50.             &#36;gray = ImageColorAllocate(&#36;im, 200,200,200);
  51.             imagefill(&#36;im,68,30,&#36;gray);
  52.    
  53.           //将四位整数验证码绘入图片
  54.             imagestring(&#36;im, 5, 10, 8, &#36;HTTP_GET_VARS[';authnum';], &#36;black);
  55.    
  56.             for(&#36;i=0;&#36;i<50;&#36;i++)  //加入干扰象素
  57.             {
  58.                 imagesetpixel(&#36;im, rand()%70 , rand()%30 , &#36;black);
  59.             }
  60.    
  61.             ImagePNG(&#36;im);
  62.             ImageDestroy(&#36;im);
  63.         ?>
复制代码
本文程序在Apache 2.0.45 + PHP 4.3.1环境下运行通过。
   
        上文只是对验证码功能的一个简单实现,并没有考虑商用安全性问题。如果要增强安全性,将此功能投入商业应用,则可以通过以下几个步骤实现:
   
        1. 启用Session。
        2. authnum在authimg.php中生成,并计算md5sum,存入session。
        3.
  authpage.php将authinput计算md5sum后,与session中的authnum(md5sum)对比得出验证结果。
   
   
        本站注:作者使用了简单的代码实现了很酷的功能。不过在添加干扰像素时的效果不是太好,大家可以看一下雨声论坛登录时的效验码(http://ror.cn/perl/ut/user_login.cgi),偶把第二段代码稍改了一下,生成了与其类似的效果。
   
        修改后的代码如下:
  1.         <?php
  2.         /*
  3.         *  Filename: authimg.php
  4.         *  Author:  hutuworm
  5.         *  Date:   2003-04-28
  6.         *  @Copyleft hutuworm.org
  7.         */
  8.         //生成验证码图片
  9.         Header("Content-type: image/PNG"); 
  10.         srand((double)microtime()*1000000);
  11.         &#36;im = imagecreate(62,20);
  12.         &#36;black = ImageColorAllocate(&#36;im, 0,0,0);
  13.         &#36;white = ImageColorAllocate(&#36;im, 255,255,255);
  14.         &#36;gray = ImageColorAllocate(&#36;im, 200,200,200);
  15.         imagefill(&#36;im,68,30,&#36;gray);
  16.         while((&#36;authnum=rand()%100000)<10000);
  17.         //将四位整数验证码绘入图片
  18.         imagestring(&#36;im, 5, 10, 3, &#36;authnum, &#36;black);
  19.         for(&#36;i=0;&#36;i<200;&#36;i++)  //加入干扰象素
  20.         {
  21.           &#36;randcolor =
  22.         ImageColorallocate(&#36;im,rand(0,255),rand(0,255),rand(0,255));
  23.           imagesetpixel(&#36;im, rand()%70 , rand()%30 , &#36;randcolor);
  24.         }
  25.         ImagePNG(&#36;im);
  26.         ImageDestroy(&#36;im);
  27.         ?>
复制代码
显示结果如下图:

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