简单验证码封装类

看到了一篇文章PHP面向对象(OOP)——自封装<验证码类> 写得不错,就拿来模仿学习。感谢作者 @Demoer

veriy_code.jpg
  1. 复制代码作者代码运行出现了错误,仔细检查了一下,发现个别手误,
      //开始填充
      private function fillBg
      {
        imagefill($this->res,0,0$this->setDarkColor());
      }
少了()
      //开始填充
      private function fillBg()
      {
        imagefill($this->res,0,0$this->setDarkColor());
      }

以及个别处的 this , 漏了$ 符号。

  1. 修改完毕之后, 网页可以正常打开生成的验证码图片。
    但是由于,想在表单处,如注册页面加入,却没办法直接插入。作者代码是直接生成了图片输出来。经和作者交流,得到一些建议。
    a) 将实例化写在 封装好的类 verify.class.php 内 , 然后 在其他页面通过
    <img src= 'verify.class.php' />

实现图片生成在指定位置, 此方法可行,但是如何验证? 再次咨询作者,得到建议是可以将生成验证码的字符串放到session里面,其他页面可以通过获取用户输入和session内存储的验证码比较。方法可行,但是个人觉得还是不在类内实例对象。于是,本人对代码进行了简单修改。

  1. 想法:通过图片转成二进制,然后在html 页面指定位置渲染。
    格式如下:
data:[<MIME-type>][;charset=<encoding>][;base64],<data>

如:

base64.png

于是修改了两个方法
作者原来的方法是

  //输出图片资源
    private function outPutImg()
    {
        //header('Content-type:image/图片类型')
        header('Content-type:image/'.$this->imgType);
        //根据图片类型,调用不同的方法输出图片            
        //imagepng($img)/imagejpg($img)
        $func = 'image'.$this->imgType;
        $func($this->res);
    }
    //开始准备生成图片
    /*方法名:show()
    *功能  :调用生成图片的所有方法
    */
    public function show()
    {
        $this->createImg();//创建图片资源
        $this->fillBg();   //填充背景颜色
        $this->fillPix();  //填充干扰点
        $this->fillArc();  //填充干扰弧线
        $this->writeFont();//写字
        $this->outPutImg();//输出图片
    }

本人将 outPutImg() 方法改成 outPutImgStr() 返回 二进制的字符串。 在show() 方法内 返回 从outPutImgStr() 得到的二进制字符串。

private function outPutImgStr()
    {
        //header('Content-type:image/图片类型')
        //header('Content-type:image/'.$this->imgType);
        ob_start();
        //根据图片类型,调用不同的方法输出图片            
        //imagepng($img)/imagejpg($img)
        $func = 'image'.$this->imgType;
        $func($this->res);
        
        // Capture the output
        $imagedata = ob_get_contents();
        // Clear the output buffer
        ob_end_clean();
        //$base64 = base64_encode($imagedata);
        $base64 = 'data:image/' . $this->imgType . ';base64,' . base64_encode($imagedata);

        return $base64;
    }

//开始准备生成图片
    /*方法名:show()
    *功能  :调用生成图片的所有方法
    */
    public function show()
    {
        $this->createImg();//创建图片资源
        $this->fillBg();   //填充背景颜色
        $this->fillPix();  //填充干扰点
        $this->fillArc();  //填充干扰弧线
        $this->writeFont();//写字
        return $this->outPutImgStr();//输出图片
    }

修改后完整代码如下:
verify.class.php

//verify.class.php
<?php

//创建一个名为Verify的验证码类
class Verify
{    
    /*
    定义验证码属性      
    */
    //图片的宽度
    public  $width;
    //图片的高度
    public  $height;
    //私有化验证码字符串,避免生成后被修改    
    private $verifyCode;
    //存储验证码字符的个数
    public  $verifyNums;
    //存储验证码的字符类型 1->纯数字 2->纯字母 3->混合类
    public  $verifyType;
    //背景颜色
    public  $bgColor;
    //文字颜色
    public  $fontColor;
    //验证码的图片类型jpg,png,bmp……
    public  $imgType;

    //图片资源
    private  $res;

    /*
    功能
    */
    //功能函数,初始化一些可以被初始化的参数
    public function __construct($width = 100,$height = 50,$imgType = 'jpg',$verifyNums = 4,$verifyType = 1)
    {
        $this->width       = $width;
        $this->height     = $height;
        $this->imgType    = $imgType;
        $this->verifyNums = $verifyNums;
        $this->verifyType = $verifyType;

        //初始化一个可以随机生成验证码的函数,将生成的验证码春初到verifyCode属性里
        $this->verifyCode = $this->createVerifyCode(); 
    }

    //随机生成验证码的函数,因为不对外公布,设置为私有的
    private function createVerifyCode()
    {
        //通过判断验证的类型来确定生成哪一种验证码
        //verifyType=1生成纯数字,为2生成纯字母,为3生成混合
          switch ($this->verifyType) {
            case 1:
        /*生成纯数字,首先使用range(0,9)生成数组
        *通过$this->verifyNums确定字符串的个数
        *使用array_rand()从数组中随机获取相应个数
        *使用join将数字拼接成字符串,存储到$str中
        */
        $str = join('',array_rand(range(0,9),$this->verifyNums));
        break;
        case 2:
        /*生成纯字母,
        *小写字母数组range('a','z')
        *大写字母数组range('A','Z')
        *合并两个数组array_merge()
        *更换键和值  array_filp()
        *随机获取数组中的特定个数的元素 array_rand()
        *拼接成字符串 implode()
        */
        $str = implode(array_rand(array_filp(array_merge(range('a','z'),range('A','Z'))),$this->verifyNums));
        break;
        case 3:
        //混合类型
        $words = str_shuffle('abcdefghjkmpopqrstuvwxyABCDEFGHIJKLMNPQRSTUVWXY3456789');
        $str = substr($words,0,$this->verifyNums);
        break;
        }
        return $str;
    }


    //开始准备生成图片
    /*方法名:show()
    *功能  :调用生成图片的所有方法
    */
    public function show()
    {
        $this->createImg();//创建图片资源
        $this->fillBg();   //填充背景颜色
        $this->fillPix();  //填充干扰点
        $this->fillArc();  //填充干扰弧线
        $this->writeFont();//写字
        return $this->outPutImgStr();//输出图片
    }

    //创建图片资源:imagecreatetruecolor($width,$height)
    private function createImg()
    {
        $this->res = imagecreatetruecolor($this->width,$this->height);
    }

    //填充背景颜色:imagefill($res,$x,$y,$color)
    //随机生成深色--->imagecolorallocate($res,$r,$g,$b)
    private function setDarkColor()
    {
      return imagecolorallocate($this->res,mt_rand(130,255),mt_rand(130,255),mt_rand(130,255));
    }    
    //随机生成浅色
    private function setLightColor()
    {
      return imagecolorallocate($this->res,mt_rand(0,130),mt_rand(0,130),mt_rand(0,130));
    }
    //开始填充
    private function fillBg()
    {
      imagefill($this->res,0,0,$this->setDarkColor());
    }

    //随机生成干扰点-->imagesetpixel
    private function fillPix()
    {
        //计算产生多少个干扰点,这里设置每20个像素产生一个
        $num = ceil(($this->width * $this->height) / 20);
        for($i = 0; $i < $num; $i++){
            imagesetpixel($this->res,mt_rand(0,$this->width),mt_rand(0,$this->height),$this->setDarkColor());
        }
    }

    //随机画10条弧线->imagearc()
    private function fillArc()
    {
      for($i = 0;$i < 10;$i++){
            imagearc($this->res,
            mt_rand(10,$this->width-10),
            mt_rand(5,$this->height-5),
            mt_rand(0,$this->width),
            mt_rand(0,$this->height),
            mt_rand(0,180),
            mt_rand(181,360),
            $this->setDarkColor());
      }
    }

    /*在画布上写文字
    *根据字符的个数,将画布横向分成相应的块
    $every = ceil($this->width/$this->verifyNums);
    *每一个小块的随机位置画上对应的字符
    imagechar();
    */

    private function writeFont()
    {
        $every = ceil($this->width / $this->verifyNums);
        for($i = 0;$i < $this->verifyNums;$i++){
            $x = mt_rand(($every * $i) + 5,$every * ($i + 1) - 5);
            $y = mt_rand(5,$this->height - 10);

            imagechar($this->res,6,$x,$y,$this->verifyCode[$i],$this->setLightColor());
        }
    }

    //输出图片资源

    private function outPutImgStr()
    {
        //header('Content-type:image/图片类型')
        //header('Content-type:image/'.$this->imgType);
        ob_start();
        //根据图片类型,调用不同的方法输出图片            
        //imagepng($img)/imagejpg($img)
        $func = 'image'.$this->imgType;
        $func($this->res);
        
        // Capture the output
        $imagedata = ob_get_contents();
        // Clear the output buffer
        ob_end_clean();
        //$base64 = base64_encode($imagedata);
        $base64 = 'data:image/' . $this->imgType . ';base64,' . base64_encode($imagedata);

        return $base64;
    }

    //设置验证码字符只能调用,不能修改,用来验证验证码书否输入正确
    public function __get($name){
      if($name = 'verifyCode'){
        return $this->verifyCode;
      }
    }

    //析构方法,自动销毁图片资源
    public function __destruct()
    {
      imagedestroy($this->res);
    }
}

?>

项目目录结构
verifyCode
-- verify.class.php
-- index.php

index.php

<?php

    //引用
    include('verify.class.php');
    //实例化
    $verify = new Verify(100,40,'png',4,3);
    //得到的二进制图片
    $a= $verify->show();
    //得到生成的验证码字符串,后面用于比较
    $b= $verify->__get('verifyCode');

?>

<html>
<head>
    
    <style type="text/css">
        .here{
            width:500px;
            height:500px;
            margin: 50px auto;
        }
        .here > input, image{
            margin: 20px;
        }
        .here > input{
            height: 40px;
        }
        .here > img{
            margin-left: 60px;
        }
    </style>
</head>
<body>
    <div class="here">
        name:<input type="text" name="name"> <br>
        <img src="<?php echo $a ?>"> <br>
        code:<input type="text" name="code">
        <p>correct code : <?php echo $b;?></p>
    </div>

 </body>
</html>

效果如下:

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

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,599评论 18 139
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,506评论 25 707
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,581评论 18 399
  • 本來旅行中的人,忙著享受旅途的生活 忙著用眼睛觀風景,雙腳踏土地 讓心去感受周圍,沒時間記錄文字 於是這樣的淺薄,...
    _行走中的蝸牛_阅读 199评论 0 0
  • 有些东西掉了就不要再捡了,人与人之间也根本就没有谁真正离不开谁,只有谁不珍惜谁,无论是故友还是红颜,一个转...
    南北有路阅读 874评论 1 6