一个简单安全的PHP验证码类、PHP验证码


Posted in PHP onSeptember 24, 2016

一,验证码示例

一个简单安全的PHP验证码类、PHP验证码

二,php验证码类,secoder.class.php

<?php 
/** 
* 安全验证码 
* 
* 安全的验证码要:验证码文字扭曲、旋转,使用不同字体,添加干扰码 
* 
* @author 流水孟春 <cmpan(at)qq.com> 
* @link http://labs.yulans.cn/YL_Security_Secoder 
* @link http://wiki.yulans.cn/docs/yl/security/secoder 
*/ 
class YL_Security_Secoder { 
/** 
* 验证码的session的下标 
* 
* @var string 
*/ 
//public static $seKey = 'sid.sek ey.ylans.cn'; 
public static $seKey = 'sid'; 
public static $expire = 3000; // 验证码过期时间(s) 
/** 
* 验证码中使用的字符,01IO容易混淆,建议不用 
* 
* @var string 
*/ 
public static $codeSet = '346789ABCDEFGHJKLMNPQRTUVWXY'; 
public static $fontSize = 25; // 验证码字体大小(px) 
public static $useCurve = true; // 是否画混淆曲线 
public static $useNoise = true; // 是否添加杂点 
public static $imageH = 0; // 验证码图片宽 
public static $imageL = 0; // 验证码图片长 
public static $length = 4; // 验证码位数 
public static $bg = array(243, 251, 254); // 背景 
protected static $_image = null; // 验证码图片实例 
protected static $_color = null; // 验证码字体颜色 
/** 
* 输出验证码并把验证码的值保存的session中 
* 验证码保存到session的格式为: $_SESSION[self::$seKey] = array('code' => '验证码值', 'time' => '验证码创建时间'); 
*/ 
public static function entry() { 
// 图片宽(px) 
self::$imageL || self::$imageL = self::$length * self::$fontSize * 1.5 + self::$fontSize*1.5; 
// 图片高(px) 
self::$imageH || self::$imageH = self::$fontSize * 2; 
// 建立一幅 self::$imageL x self::$imageH 的图像 
self::$_image = imagecreate(self::$imageL, self::$imageH); 
// 设置背景 
imagecolorallocate(self::$_image, self::$bg[0], self::$bg[1], self::$bg[2]); 
// 验证码字体随机颜色 
self::$_color = imagecolorallocate(self::$_image, mt_rand(1,120), mt_rand(1,120), mt_rand(1,120)); 
// 验证码使用随机字体 
//$ttf = dirname(__FILE__) . '/ttfs/' . mt_rand(1, 20) . '.ttf'; 4 
$ttf = dirname(__FILE__) . '/ttfs/4.ttf'; 
if (self::$useNoise) { 
// 绘杂点 
self::_writeNoise(); 
} 
if (self::$useCurve) { 
// 绘干扰线 
self::_writeCurve(); 
} 
// 绘验证码 
$code = array(); // 验证码 
$codeNX = 0; // 验证码第N个字符的左边距 
for ($i = 0; $i<self::$length; $i++) { 
$code[$i] = self::$codeSet[mt_rand(0, 27)]; 
$codeNX += mt_rand(self::$fontSize*1.2, self::$fontSize*1.6); 
// 写一个验证码字符 
imagettftext(self::$_image, self::$fontSize, mt_rand(-40, 70), $codeNX, self::$fontSize*1.5, self::$_color, $ttf, $code[$i]); 
} 
// 保存验证码 
isset($_SESSION) || session_start(); 
$_SESSION[self::$seKey]['code'] = join('', $code); // 把校验码保存到session 
$_SESSION[self::$seKey]['time'] = time(); // 验证码创建时间 
header('Cache-Control: private, max-age=0, no-store, no-cache, must-revalidate'); 
header('Cache-Control: post-check=0, pre-check=0', false); 
header('Pragma: no-cache'); 
header("content-type: image/png"); 
// 输出图像 
imagepng(self::$_image); 
imagedestroy(self::$_image); 
} 
/** 
* 画一条由两条连在一起构成的随机正弦函数曲线作干扰线(你可以改成更帅的曲线函数) 
* 
* 高中的数学公式咋都忘了涅,写出来 
* 正弦型函数解析式:y=Asin(ωx+φ)+b 
* 各常数值对函数图像的影响: 
* A:决定峰值(即纵向拉伸压缩的倍数) 
* b:表示波形在Y轴的位置关系或纵向移动距离(上加下减) 
* φ:决定波形与X轴位置关系或横向移动距离(左加右减) 
* ω:决定周期(最小正周期T=2π/?ω?) 
* 
*/ 
protected static function _writeCurve() { 
$A = mt_rand(1, self::$imageH/2); // 振幅 
$b = mt_rand(-self::$imageH/4, self::$imageH/4); // Y轴方向偏移量 
$f = mt_rand(-self::$imageH/4, self::$imageH/4); // X轴方向偏移量 
$T = mt_rand(self::$imageH*1.5, self::$imageL*2); // 周期 
$w = (2* M_PI)/$T; 
$px1 = 0; // 曲线横坐标起始位置 
$px2 = mt_rand(self::$imageL/2, self::$imageL * 0.667); // 曲线横坐标结束位置 
for ($px=$px1; $px<=$px2; $px=$px+ 0.9) { 
if ($w!=0) { 
$py = $A * sin($w*$px + $f)+ $b + self::$imageH/2; // y = Asin(ωx+φ) + b 
$i = (int) ((self::$fontSize - 6)/4); 
while ($i > 0) { 
imagesetpixel(self::$_image, $px + $i, $py + $i, self::$_color); // 这里画像素点比imagettftext和imagestring性能要好很多 
$i--; 
} 
} 
} 
$A = mt_rand(1, self::$imageH/2); // 振幅 
$f = mt_rand(-self::$imageH/4, self::$imageH/4); // X轴方向偏移量 
$T = mt_rand(self::$imageH*1.5, self::$imageL*2); // 周期 
$w = (2* M_PI)/$T; 
$b = $py - $A * sin($w*$px + $f) - self::$imageH/2; 
$px1 = $px2; 
$px2 = self::$imageL; 
for ($px=$px1; $px<=$px2; $px=$px+ 0.9) { 
if ($w!=0) { 
$py = $A * sin($w*$px + $f)+ $b + self::$imageH/2; // y = Asin(ωx+φ) + b 
$i = (int) ((self::$fontSize - 8)/4); 
while ($i > 0) { 
imagesetpixel(self::$_image, $px + $i, $py + $i, self::$_color); // 这里(while)循环画像素点比imagettftext和imagestring用字体大小一次画出(不用这while循环)性能要好很多 
$i--; 
} 
} 
} 
} 
/** 
* 画杂点 
* 往图片上写不同颜色的字母或数字 
*/ 
protected static function _writeNoise() { 
for($i = 0; $i < 10; $i++){ 
//杂点颜色 
$noiseColor = imagecolorallocate( 
self::$_image, 
mt_rand(150,225), 
mt_rand(150,225), 
mt_rand(150,225) 
); 
for($j = 0; $j < 5; $j++) { 
// 绘杂点 
imagestring( 
self::$_image, 
5, 
mt_rand(-10, self::$imageL), 
mt_rand(-10, self::$imageH), 
self::$codeSet[mt_rand(0, 27)], // 杂点文本为随机的字母或数字 
$noiseColor 
); 
} 
} 
} 
/** 
* 验证验证码是否正确 
* 
* @param string $code 用户验证码 
* @param bool 用户验证码是否正确 
*/ 
public static function check($code) { 
isset($_SESSION) || session_start(); 
// 验证码不能为空 
if(empty($code) || empty($_SESSION[self::$seKey])) { 
//echo $_SESSION[self::$seKey]['code'].'1'; 
return false; 
} 
// session 过期 
if(time() - $_SESSION[self::$seKey]['time'] > self::$expire) { 
unset($_SESSION[self::$seKey]); 
//echo $_SESSION[self::$seKey]['code'].'2'; 
return false; 
//return 0; 
} 
// if($code == $_SESSION[self::$seKey]['code']) { 
if(strtoupper($code) == $_SESSION[self::$seKey]['code']) { //不区分大小写比较 
//echo $_SESSION[self::$seKey]['code'].'3'; 
return true; 
} 
//echo $_SESSION[self::$seKey]['code'].'4'; 
return false; 
} 
} 
// useage 
/* 
YL_Security_Secoder::$useNoise = false; // 要更安全的话改成true 
YL_Security_Secoder::$useCurve = true; 
YL_Security_Secoder::entry(); 
*/ 
/* 
// 验证验证码 
if (!YL_Security_Secoder::check(@$_POST['secode'])) { 
print 'error secode'; 
} 
*/

三,调用方法

1,显示验证码页面code.php

<?php 
session_start(); 
require 'secoder.class.php'; //先把类包含进来,实际路径根据实际情况进行修改。 
$vcode = new YL_Security_Secoder(); //实例化一个对象 
$vcode->entry(); 
?>

2,检查验证码是否正确

<?php 
session_start(); 
require 'secoder.class.php'; //先把类包含进来,实际路径根据实际情况进行修改。 
$vcode = new YL_Security_Secoder(); //实例化一个对象 
//$vcode->entry(); 
$code = $_GET['code']; 
echo $vcode->check($code); 
//$_SESSION['code'] = $vc->getCode();//验证码保存到SESSION中 
?>

3,验证码输入框调用页面

<img id="messageImg" src='images/tishis2.gif' width='16' height='16'> 单击图片重新获取验证码<br> 
<a href="#"><img src="code.php" onclick="javascript:this.src='code.php?tm='+Math.random();" />

以上所述是小编给大家介绍的一个简单安全的PHP验证码类、PHP验证码,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对三水点靠木网站的支持!

PHP 相关文章推荐
支持oicq头像的留言簿(一)
Oct 09 PHP
PHP5.2中date()函数显示时间与北京时间相差8小时的解决办法
May 28 PHP
php快速url重写 更新版[需php 5.30以上]
Apr 20 PHP
PHP中的插件机制原理和实例
Jul 08 PHP
PHP实现设计模式中的抽象工厂模式详解
Oct 11 PHP
php递归删除目录与文件的方法
Jan 30 PHP
php读取文件内容到数组的方法
Mar 16 PHP
详解PHP序列化反序列化的方法
Oct 27 PHP
深入理解PHP内核(一)
Nov 10 PHP
PHP封装的字符串加密解密函数
Dec 18 PHP
CI框架简单邮件发送类实例
May 18 PHP
PHP实现的一致性Hash算法详解【分布式算法】
Mar 31 PHP
jquery不支持toggle()高(新)版本的问题解决
Sep 24 #PHP
php getcwd与dirname(__FILE__)区别详解
Sep 24 #PHP
mysql desc(DESCRIBE)命令实例讲解
Sep 24 #PHP
mysql alter table命令修改表结构实例详解
Sep 24 #PHP
Ajax和PHP正则表达式验证表单及验证码
Sep 24 #PHP
mysql查找删除重复数据并只保留一条实例详解
Sep 24 #PHP
php源码 fsockopen获取网页内容实例详解
Sep 24 #PHP
You might like
php&amp;java(一)
2006/10/09 PHP
php&amp;java(二)
2006/10/09 PHP
thinkphp中的url跳转用法分析
2016/07/12 PHP
Yii1.1中通过Sql查询进行的分页操作方法
2017/03/16 PHP
js解析与序列化json数据(二)序列化探讨
2013/02/01 Javascript
在JavaScript中typeof的用途介绍
2013/04/11 Javascript
javascript中的原型链深入理解
2014/02/24 Javascript
javascript写的一个模拟阅读小说的程序
2014/04/04 Javascript
js判断登录与否并确定跳转页面的方法
2015/01/30 Javascript
javascript实现日期按月份加减
2015/05/15 Javascript
Bootstrap多级导航栏(级联导航)的实现代码
2016/03/08 Javascript
用jquery获取自定义的标签属性的值简单实例
2016/09/17 Javascript
javascript的函数劫持浅析
2016/09/26 Javascript
详解nodeJS之二进制buffer对象
2017/06/03 NodeJs
剖析Angular Component的源码示例
2018/03/23 Javascript
Vue引入jquery实现平滑滚动到指定位置
2018/05/09 jQuery
傻瓜式vuex语法糖kiss-vuex整理
2018/12/21 Javascript
使用 vue 实现灭霸打响指英雄消失的效果附demo
2019/05/06 Javascript
微信小程序自定义modal弹窗组件的方法详解
2020/12/20 Javascript
从源码角度来回答keep-alive组件的缓存原理
2021/01/18 Javascript
王纯业的Python学习笔记 下载
2007/02/10 Python
Python 第一步 hello world
2009/09/25 Python
Python 遍历子文件和所有子文件夹的代码实例
2016/12/21 Python
python MNIST手写识别数据调用API的方法
2018/08/08 Python
python调用opencv实现猫脸检测功能
2019/01/15 Python
Django中ORM外键和表的关系详解
2019/05/20 Python
Python可变对象与不可变对象原理解析
2020/02/25 Python
日本最大的购物网站:日本乐天市场(Rakuten Ichiba)
2020/11/04 全球购物
2014年社区学雷锋活动总结
2014/03/09 职场文书
旅游安全协议书
2014/04/21 职场文书
公司市场专员岗位职责
2014/06/29 职场文书
MySQL 逻辑备份与恢复测试的相关总结
2021/05/14 MySQL
Go语言实现Base64、Base58编码与解码
2021/07/26 Golang
Java 在生活中的 10 大应用
2021/11/02 Java/Android
抖音动画片,皮皮虾,《治愈系》动画在用这首REMIX作为背景音乐,Anak ,The last world with you完整版
2022/03/16 杂记
《进击的巨人》新联动CM 兵长强势出击兽巨人
2022/04/05 日漫