【www.bbyears.com--php常用代码】
PHP解析验证码类
1.开始
在网上看到使用PHP写的ValidateCode生成验证码类,感觉不错,特拿来分析学习一下。
2.类图
3.验证码类部分代码
3.1 定义变量
1 2 3 4 5 6 7 8 9 10 11 //随机因子 private$charset="abcdefghjkmnprstuvwxyzABCDEFGJKMNPRSTUVWXYZ23456789"; private$code; private$codeLen= 4; private$width= 130; private$heigh= 50; private$img;//图像 private$font;//字体 private$fontsize= 20;$charset 是随机因子,这里是去掉了几个不容易区分的字符,如字母"i,l,o,q",数字"0,1"。有必要可以加入一些中文或其他字符或算式等。
$codeLen表示验证码长度,常见4位。
3.2构造函数,设置验证码字体,生成一个真彩色图像img
1 2 3 4 5 publicfunction__construct() { $this->font = ROOT_PATH."/font/Chowderhead.ttf"; $this->img = imagecreatetruecolor($this->width,$this->heigh); }3.3从随机因子中随机抽取4个字符,作为$code验证码.
1 2 3 4 5 6 7 8 //生成随机码 privatefunctioncreateCode() { $_len=strlen($this->charset) - 1; for($i= 0;$i<$this->codeLen;$i++) { $this->code .=$this->charset[mt_rand(0,$_len)]; } }3.4生成验证码背景色.
1 2 3 4 5 6 7 //生成背景 privatefunctioncreateBg() { $color= imagecolorallocate($this->img, mt_rand(157, 255), mt_rand(157, 255), mt_rand(157, 255)); imagefilledrectangle($this->img, 0,$this->heigh,$this->width, 0,$color); }其中mt_rand(157, 255),目的是随机取比较浅的颜色。
3.5在图像上生成文字.
1 2 3 4 5 6 7 8 9 10 //生成文字 privatefunctioncreateFont() { $_x=$this->width /$this->codeLen; $_y=$this->heigh / 2; for($i= 0;$i<$this->codeLen;$i++) { $color= imagecolorallocate($this->img, mt_rand(0, 156), mt_rand(0, 156), mt_rand(0, 156)); imagettftext($this->img,$this->fontsize, mt_rand(-30, 30),$_x*$i+ mt_rand(3, 5),$_y+ mt_rand(2, 4),$color,$this->font,$this->code[$i]); } }在图像上生成验证码文字,主要考虑文字在图像上的位置和每一个文字颜色。
控制第n个文字的x轴位置 = (图像宽度 / 验证码长度) * (n-1) + 随机的偏移数; 其中n = {d1....n}
控制第n个文字的y轴位置 = 图像高度 / 2 + 随机的偏移数;
mt_rand(0, 156) 随机取文字颜色,0-156目的是取比较深的颜色。
mt_rand(-30, 30) 随机的文字旋转。
3.6在图像上生成线条和雪花
1 2 3 4 5 6 7 8 9 10 11 12 //生成线条,雪花 privatefunctioncreateLine() { for($i= 0;$i< 15;$i++) { $color= imagecolorallocate($this->img, mt_rand(0, 156), mt_rand(0, 156), mt_rand(0, 156)); imageline($this->img, mt_rand(0,$this->width), mt_rand(0,$this->heigh), mt_rand(0,$this->width), mt_rand(0,$this->heigh),$color); } for($i= 0;$i< 150;$i++) { $color= imagecolorallocate($this->img, mt_rand(200, 255), mt_rand(200, 255), mt_rand(200, 255)); imagestring($this->img, mt_rand(1, 5), mt_rand(0,$this->width), mt_rand(0,$this->heigh),"#",$color); } }画线条的时候,取比较深的颜色值,而画雪花的时候取比较淡的颜色值,目的是尽可能的不影响人眼识别验证码,又能干扰自动识别验证码机制。
3.7对外生成验证码图像,供外部调用。
1 2 3 4 5 6 7 8 9 10 //对外生成 publicfunctiondoImg() { $this->createBg(); //1.创建验证码背景 $this->createCode(); //2.生成随机码 $this->createLine(); //3.生成线条和雪花 $this->createFont(); //4.生成文字 $this->outPut(); //5.输出验证码图像 }3.8完整代码:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 font = ROOT_PATH."/font/Chowderhead.ttf"; $this->img = imagecreatetruecolor($this->width,$this->heigh); } //生成随机码 privatefunctioncreateCode() { $_len=strlen($this->charset) - 1; for($i= 0;$i<$this->codeLen;$i++) { $this->code .=$this->charset[mt_rand(0,$_len)]; } } //生成背景 privatefunctioncreateBg() { $color= imagecolorallocate($this->img, mt_rand(157, 255), mt_rand(157, 255), mt_rand(157, 255)); imagefilledrectangle($this->img, 0,$this->heigh,$this->width, 0,$color); } //生成文字 privatefunctioncreateFont() { $_x=$this->width /$this->codeLen; $_y=$this->heigh / 2; for($i= 0;$i<$this->codeLen;$i++) { $color= imagecolorallocate($this->img, mt_rand(0, 156), mt_rand(0, 156), mt_rand(0, 156)); imagettftext($this->img,$this->fontsize, mt_rand(-30, 30),$_x*$i+ mt_rand(3, 5),$_y+ mt_rand(2, 4),$color,$this->font,$this->code[$i]); } } //生成线条,雪花 privatefunctioncreateLine() { for($i= 0;$i< 15;$i++) { $color= imagecolorallocate($this->img, mt_rand(0, 156), mt_rand(0, 156), mt_rand(0, 156)); imageline($this->img, mt_rand(0,$this->width), mt_rand(0,$this->heigh), mt_rand(0,$this->width), mt_rand(0,$this->heigh),$color); } for($i= 0;$i< 150;$i++) { $color= imagecolorallocate($this->img, mt_rand(200, 255), mt_rand(200, 255), mt_rand(200, 255)); imagestring($this->img, mt_rand(1, 5), mt_rand(0,$this->width), mt_rand(0,$this->heigh),"#",$color); } } //输出图像 privatefunctionoutPut() { header("Content-Type: image/png"); imagepng($this->img); imagedestroy($this->img); } //对外生成 publicfunctiondoImg() { $this->createBg(); //1.创建验证码背景 $this->createCode(); //2.生成随机码 $this->createLine(); //3.生成线条和雪花 $this->createFont(); //4.生成文字 $this->outPut(); //5.输出验证码图像 } //获取验证码 publicfunctiongetCode() { returnstrtolower($this->code); } }4.测试
测试代码:
1 2 3 4 5 6 7 8 9 10 11 12 13 doImg();生成验证码:
5.应用
1 2 3上面onclick代码是点击验证码图片,能自动刷新验证码。
code.php:
1 2 3 4 5 6 7 8 9 10 11 12 doImg(); $_SESSION["ValidateCode"]=$_vc->getCode();有关应用的完整代码可以从https://git.oschina.net/andywww/myTest 的CMS1.0 文件里下载。
6.小结
在独立测试过程,没发现什么问题;但应用到项目的时候,刚开始发现无法生成验证码图片,网上找了一下,有的说是在outPut()函数中,
在header("Content-Type: image/png"); 这行代码前面增加了一行ob_clean()代码,可以解决验证码这块问题。虽然此方法简单,但这可能会引起其他缓冲数据问题,因为db_clean()功能就是丢弃输出缓冲区中的内容。