PHP实现验证码制作

验证码是什么?

验证码为全自动区分计算机和人类的图灵测试的缩写,是一种区分用户是计算机和人的公共全自动程序;


学习的目的

屏蔽机器的请求,把服务提供给人,保障业务不受机器提交请求干扰;

验证码应用场景

  • 登录、注册确定提交前,做人/机器的校验;
  • 发布、回复信息前,做人/机器的校验;
    目的区分是人提交的还是机器提交的,那么怎么区分出来的呢?验证码填写正确当做是人提交的,输入错误就当做机器处理如果是机器那么不予通过;

步骤解析

1、生成底图
2、生成验证内容
3、生成验证码图片
4、校验验证码内容


技术点:

  • 底图的实现,并添加干扰元素
    依赖PHP图片处理库GD
    添加干扰元素的好处就是,有人在恶意识别这个验证码时,需要在他机器算法里识别策略里增加更多的策略来识别出这个点,这也就增加了破解难度;
  • 生成验证内容
    随机数生成,使用PHP函数mt_rand()
    随机数字+字母,需要ASCII码基础;
    随机中文,需要UTF-8编码基础
  • 验证内容保存在服务端
    使用PHP会话机制SESSION服务;
  • 验证内容的校验
    需要前端Ajax技术;

代码展示

<?php
function code() {
    session_start();
    //生成底图
    $image = imagecreatetruecolor(100,30);//生成一个默认黑色画布
    $bgcolor = imagecolorallocate($image,255,255,255);//#FFFFFF
    imagefill($image,0,0,$bgcolor);
    //生成验证内容(控制好字体大小与分布,避免字体重叠或显示不全)
    $code = '';
    for ($i = 0; $i < 4; $i++) {
        $fontsize = 6;//字体大小
        $fontcolor = imagecolorallocate($image,rand(0,120),rand(0,120),rand(0,120));//字体颜色
//        $str = 'qwertyuiopasdfghjklzxcvbnm1234567890';
//        $fontcontent = substr($str,rand(0,strlen($str)),1);//随机数字+字母
        $fontcontent = rand(0,9);//随机数字
        $code .= $fontcontent;
        $x = ($i * 100 / 4) + rand(5,10);
        $y = rand(5,10);
        imagestring($image,$fontsize,$x,$y,$fontcontent,$fontcolor);//填充内容
    }
    $_SESSION['code'] = $code;
    //增加干扰元素点或线(控制好颜色,避免喧宾夺主)
    for ($i = 0; $i < 200; $i++) {
        $pointcolor = imagecolorallocate($image,rand(50,200),rand(50,200),rand(50,200));
        imagesetpixel($image,rand(1,99),rand(1,29),$pointcolor);
    }
    for ($i = 0; $i < 3; $i++) {
        $linecolor = imagecolorallocate($image,rand(80,200),rand(80,220),rand(80,220));
        imageline($image,rand(1,99),rand(1,29),rand(1,99),rand(1,29),$linecolor);
    }
    header("Content-type:image/png");//输出图片前,必须输出图片header信息
    imagepng($image);//打印输出
    imagedestroy($image);//释放资源
}

点击刷新验证码

onclick="this.src='code.php?time='+Math.random()"

添加图片水印--文字水印

<?php
function TextWatermark($text,$dst_path,$x,$y) {
    //创建图片的实例
    $dst = imagecreatefromstring(file_get_contents($dst_path));
    //打上文字
    $font = 'simsun.ttc';//字体
    $black = imagecolorallocate($dst, 0, 0, 0);//字体颜色
    imagefttext($dst, 15, 0, $x, $y, $black, $font, $text);
    //输出图片
    list($dst_w, $dst_h, $dst_type) = getimagesize($dst_path);
    switch ($dst_type) {
        case 1://GIF
            imagegif($dst,"static/admin/upload/".time().".gif");
            return "static/admin/upload/".time().".gif";
            break;
        case 2://JPG
            imagejpeg($dst,"static/admin/upload/".time().".jpg");
            return "static/admin/upload/".time().".jpg";
            break;
        case 3://PNG
            imagepng($dst,"static/admin/upload/".time().".png");
            return "static/admin/upload/".time().".png";
            break;
        default:
            break;
    }
    imagedestroy($dst);
}

function ImgWatermark($dst_path,$src_path,$x,$y) {
    //创建图片的实例
    $dst = imagecreatefromstring(file_get_contents($dst_path));
    $src = imagecreatefromstring(file_get_contents($src_path));
    //获取水印图片的宽高
    list($src_w, $src_h) = getimagesize($src_path);
    //将水印图片复制到目标图片上,最后个参数50是设置透明度,这里实现半透明效果
    imagecopymerge($dst, $src, $x,$y, 0,0, $src_w, $src_h, 50);
    //如果水印图片本身带透明色,则使用imagecopy方法
    //imagecopy($dst, $src, 10, 10, 0, 0, $src_w, $src_h);
    //输出图片
    list($dst_w, $dst_h, $dst_type) = getimagesize($dst_path);
    switch ($dst_type) {
        case 1://GIF
            imagegif($dst,"static/admin/upload/".time().".gif");
            return "static/admin/upload/".time().".gif";
            break;
        case 2://JPG
            imagejpeg($dst,"static/admin/upload/".time().".jpg");
            return "static/admin/upload/".time().".jpg";
            break;
        case 3://PNG
            imagepng($dst,"static/admin/upload/".time().".png");
            return "static/admin/upload/".time().".png";
            break;
        default:
            break;
    }
    imagedestroy($dst);
    imagedestroy($src);
}

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 217,826评论 6 506
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,968评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 164,234评论 0 354
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,562评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,611评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,482评论 1 302
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,271评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,166评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,608评论 1 314
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,814评论 3 336
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,926评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,644评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,249评论 3 329
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,866评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,991评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,063评论 3 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,871评论 2 354

推荐阅读更多精彩内容

  • 背景 验证码就是把一串随机产品的数字动态生成一幅图片,再加上干扰元素。此时用户可以通过肉眼能识别里面的数字或者字符...
    dy2903阅读 2,093评论 0 7
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,656评论 18 139
  • Composer Repositories Composer源 Firegento - Magento模块Comp...
    零一间阅读 3,960评论 1 66
  • 已经搬家到新博客 jjyy.guru继续和大家跟分享游戏技术、编程心得、unity3d、cocos2d、emac...
    不是流汗羊阅读 2,772评论 10 72
  • 夏天没有冰淇淋 就好像方便面没有 调料包 …… 最近连续的高温天气, 分分钟热化人, 出门五分钟,流汗两小时, 这...
    小爷很乖阅读 243评论 0 0