PHP基础教程十六之使用GD库进行图形绘制

x33g5p2x  于2022-03-06 转载在 其他  
字(8.5k)|赞(0)|评价(0)|浏览(418)

前言

我们在上网登录网页的时候,绝大多数都要输入验证码,进行验证,才能登录,而PHP程序员可以根据情况自己绘制验证码,在PHP中,简单的验证码,我们可以使用PHP自带的绘图功能来绘制,并且我们也可以使用PHP的绘图对图片进行水印的增加……

PHP绘图坐标系

我们学数学的时候都要学习坐标系,这里就不在进行阐述,在编程语言中也是有坐标系的存在,但是编程语言的坐标系和数学上的坐标系是不一样的。

在PHP坐标系中,坐标原点位于左上角
,X轴为水平方向,Y轴是当前位置的垂直方向,同时坐标系的单位是像素。

像素

上面说到坐标系的坐标点是像素,像素是一个密度单位,它的大小和用户的屏幕有关系。

像素:计算机在屏幕上显示的内容都是由屏幕上的每一个像素组成的,例如,计算机显示器的分辨率是800×600,表示计算机屏幕上的每一行由800个点组成,共有600行,整个计算机屏幕共有480000个像素。

注意:像素是密度单位,厘米是长度单位。

PHP绘图

在PHP中如果要使用绘图技术,必须打开gd2扩展库。在PHP.ini文件中进行打开,打开后重启服务器。

在PHP绘图功能中的函数都是以image开头的。并且整个函数都是小写。在帮助文档中有很多函数。在我们需要的时候可以在里面找:函数参考-图像生成和处理-GD-GD和图像处理函数。我们在进行PHP绘图中,一般可以按照步骤一步步完成。

绘图的步骤

  1. 创建一个图像资源,相当于一个画布,我们可以在这个画布上绘画(相当于一张纸),创建画布方式中的一种:
$canva = imagecreatetruecolor(宽,高)
  1. 创建一个画笔,使用画笔在画布上绘画,而画笔的颜色可以用RGB进行配置。方法里面的参数是 0 到 255 的整数或者十六进制的 0x00 到 0xFF。
$paint = imagecolorallocate(画布,三原色(r,g,b));
  1. 绘制图形,这里只是绘制一条直线,也可以绘制其他任意的图像。
imageline(画布,startX,startY,endX,endY,画笔);
  1. 绘制完成,可以输出到网页上。也可以保存在某个目录下。
  2. 销毁绘图的资源,及时的释放内存。
imagedestroy(画布);

我们可以按照上面的步骤进行绘制不同的图形,在进行绘制图形的时候,我们可以进行不同的操作进行图形的拼接,同时我们也可以创建不同的画笔,进行不同颜色的绘制。

绘制不同的图形

我们可以根据上面的步骤进行不同图案的绘制。只不过是绘制不同的图形需要不同的函数。函数的参数可以在帮助文档中进行查询。

<?php

    //创建一个画布
    $canvas = imagecreatetruecolor(300, 300);
    //创建一个画笔
    $paint = imagecolorallocate($canvas, 255, 0, 0);

    //绘制不同的图形,可以修改函数进行不同图形的绘制
    //绘制一条直线
    imageline($canvas, 10, 10, 50, 50, $paint);
    //绘制一个矩形
    imagerectangle($canvas, 10, 100, 30, 120, $paint);

    //绘制一个椭圆
    imageellipse($canvas, 100, 121, 100, 50, $paint);
    //绘制一个圆,也是上面的函数,不过是把宽和高写成一样的就行了。
    imageellipse($canvas,200,100,50,50,$paint);
    //绘制弧形
    imagearc($canvas, 200,50,40, 40, 0, 90, $paint);
    //画一个填充的圆形
    imagefilledellipse($canvas, 250, 230, 60, 60, $paint);
    //输出到浏览器
    header("content-type:image/png");
    //以png的形式输出到浏览器上
    imagepng($canvas);

    //销毁图形,释放内存
    imagedestroy($canvas);

上面的代码使用不同的函数创建不同的图形,但是我们在上面的代码中只有一个对图像进行填充,比如我们画一个圆里面填充上颜色。可以使用imagefilledellipse()函数,这个函数只是比imageellipse()函数多了一个filled,里面的参数还是一样的。在绘制图形中,如果要填充图形,一般是在image后面添加filled这个字母就可以了。

PHP绘制图形是还可以使用现成的图片进行绘制,使用这种方式可以理解为画布开始就有图案了,我们只是在图案上进行绘制。

<?php
    //使用现有的图形进行绘制
    $canvas = imagecreatefromjpeg('Desert.jpg');
    //创建画笔
    $paint_red = imagecolorallocate($canvas,255,0,0);
    //在创建一个蓝色的笔。
    $paint_blue = imagecolorallocate($canvas,0,0,255);
    //绘制一个填充的弧形,最后一个参数是连接的样式。
    imagefilledarc($canvas, 100, 100, 100, 100, 0, 180, $paint_blue,IMG_ARC_PIE 
);  

    //输出到浏览器
    header("content-type:image/png");
    //以png的形式输出到浏览器上
    imagepng($canvas);

    //销毁图形,释放内存
    imagedestroy($canvas);

从结果中可以看到在图形的左上角画了一个填充的半圆形。

当然,如果我们不想修改原来的图形,可以使用copy方法把现有的图形复制下来。这样你对图形的修改就不会形象到原来的图形。

<?php
    //创建一个空白的画布
    $canvas = imagecreatetruecolor(500,500);
    //创建一个图片
    $image = imagecreatefromjpeg('Desert.jpg');
    //使用函数获取图片的信息,里面包括了图片的宽高。
    $image_info = getimagesize('Desert.jpg');
    //使用imagecopy()方法进行拷贝
    //第一个参数是把图形拷贝到那个画布,第二个参数是从哪里拷贝,第三,四个参数是拷贝到的画布从哪里开始,第五六七八个参数是被拷贝的图形从哪里开始,宽和高是多少
    imagecopy($canvas, $image, 0, 0, 0, 0, $image_info[0], $image_info[1]);

    //在拷贝的图形上绘制一个填充的矩形。

    $paint = imagecolorallocate($canvas,0,255,0);
    imagefilledrectangle($canvas, 100, 100, 300, 300, $paint);
    //输出到浏览器
    header("content-type:image/png");
    //以png的形式输出到浏览器上
    imagepng($canvas);

    //销毁图形,释放内存
    imagedestroy($canvas);

上面使用imagecopy()函数进行,我们在使用这个函数的时候需要注意里面的参数。

在开发中我们会遇到在一张图片上打印上水印。这里我们有两个函数都可以进行imagestring()函数和imagettftext()函数,但是这两个函数是有不同的,imagestring()函数不支持中文,并且它支持的字体太少,只有五种字体,imagettftext()函数可以很好的输出中文,并且它的字体可以自由的设置,但是这个函数需要一个字体文件。

<?php

    //创建一个空白的画布
    $canvas = imagecreatetruecolor(500,500);
    //创建一个图片
    $image = imagecreatefromjpeg('Desert.jpg');
    //使用函数获取图片的信息,里面包括了图片的宽高。
    $image_info = getimagesize('Desert.jpg');
    //使用imagecopy()方法进行拷贝
    imagecopy($canvas, $image, 0, 0, 0, 0, $image_info[0], $image_info[1]);
    //绘制水印的
    $paint = imagecolorallocate($canvas, 255, 0, 0);
    //使用imagestring()函数
    $str = 'lijiafei';
    //这个函数第二个参数是设置字体的,只能是1-5,
    imagestring($canvas, 5, 100, 300, $str, $paint);

    //使用imagettftext()函数
    $text = '这是一张中文的水印';
    imagettftext($canvas, 30, 0, 100, 200, $paint, 'STXINGKA.TTF', $text);

    header("content-type:image/png");
    //以png的形式输出到浏览器上
    imagepng($canvas);

    //销毁图形,释放内存
    imagedestroy($canvas);

在上面的代码中我们可以看到使用两个函数进行水印的打印。

  1. imagestring(resource $image , int $font , int $x , int $y , string $s , int $col )

  2. 第一个参数可以理解为是一个画布,

  3. 第二个参数是水印的字体大小,这里只支持1-5五种字体。

  4. 第三、四个参数是水印位于图片的那个位置,坐标位于字体的左上角。

  5. 第五个参数是需要打印的字体。

  6. 第六个参数字体的颜色。

  7. imagettftext(resource $image,float $size,float $angle,int $x,int $y,int $color,string $fontfile,string $text)这个函数的参数有点多,但是也不难理解。

  8. 第一个参数是一个画布

  9. 第二个参数是字体的大小,这里的字体的大小我们可以自由的设置,数字越大,字体越大。

  10. 第三个参数是字体倾斜的度数,按照逆时针的顺序进行倾斜。

  11. 第四、五个参数是字体位于图片的位置,这里的坐标是从字体的左下角计算的,和上面的不同。

  12. 第六个参数是字体的颜色。

  13. 第七个参数是一个字体文件的名字,在我们的C:\Windows\Fonts下有很多字体,我们可以选择一种字体,字体文件我们需要粘贴到当前文件下面才能生效。

  14. 第八个参数是需要打印的字。

绘图案例

上面我们介绍了大多数函数的介绍,这里我们做一个案列,这个案例说白了就是利用不同的函数进行拼接图形。

代码:

<?php
    //创建一个画布
    $canvas = imagecreatetruecolor(500, 500);
    //蓝色的笔
    $blue = imagecolorallocate($canvas, 0, 0, 255);
    //白的的笔
    $white = imagecolorallocate($canvas,255,255,255);
    //黑色的笔
    $black = imagecolorallocate($canvas,0,0,0);
    //红色的笔
    $red = imagecolorallocate($canvas,255,0,0);
    //画头,画一个填充颜色是蓝色的圆。
    imagefilledellipse($canvas, 150, 150, 130, 105, $blue);
    //画脸,画一个填充颜色是白色的圆。但是圆的原点比头的原点要往下一点。
    imagefilledellipse($canvas, 150, 160, 110, 90, $white);
    //画两个眼睛
    imageellipse($canvas, 140, 130, 20, 30, $black);
    imageellipse($canvas, 160, 130, 20, 30, $black);
    //画两个眼珠
    imagefilledellipse($canvas, 140, 130, 5, 5, $black);
    imagefilledellipse($canvas, 160, 130, 5, 5, $black);
    //画鼻子
    imagefilledellipse($canvas, 150, 150, 10, 10, $red);
    //画鼻子下面的一条线
    imageline($canvas,150,155,150,165,$black);
    //画嘴
    imagefilledarc($canvas, 150,165, 75, 60, 0, 180, $red,IMG_ARC_PIE);
    //画嘴的两个角
    imagearc($canvas, 114,156, 20, 20, 90, 270, $black);
    imageline($canvas,114,165,187,165,$black);
    imagearc($canvas, 187,156, 20, 20, 270, 90, $black);
    //把嘴巴画成黑色
    imagearc($canvas, 150,165, 75, 60, 0, 180, $black);

    header("content-type:image/png");

    imagepng($canvas);
    imagedestroy($canvas);

利用绘图制作验证码

我们在开头提到过,我们可以利用绘图技术来进行网站验证码的制作,而验证码的生成可以看成在一张图片上把随机产生的字体打印在图片上。同时在图片上有一些点进行视线的骚扰,这里我们可以创建一个类,然后通过不同的方法进行不同的操作。

  1. 创建一个方法进行随机数的生成。
/*
生成随机码
 */
private function mkCode(){
    //通过类的参数获取需要的随机数的个数。这个值可以自由的指定
    $len = $this -> _len;
    //我们生成的随机数的字母喝数字就是在这里面进行随机生成。
    $str = 'ABCDEFGHIGKLMNOPQRST1234567890';
    $code = '';
    //通过循环的生成随机数进行获取
    for($i = 0; $i < $len; $i++){
        //生成随机数
        $j = mt_rand(0,strlen($str)-1);
        //把随机生成的随机数拼接起来。
        $code .= $str[$j];
    }
    //把生成的随机数,保存在session中,便于当我们输入验证码是验证是否正确。
    @session_start();
    $_SESSION['code'] = $code;
    return $code;
}

我们在生成随机码的时候,可以指定生成的随机数的长度,默认是4个。通过mt_rand(参数1,参数2)函数生成不同的数字。这个函数返回一个参数1和参数2之间(包括这两个数)的int类型的数字。

  1. 上面我们已经生成一个随机的验证码。那么接下来就可以把随机生成的随机数打印到一个画布上。
//生成验证码
    public function makeImage(){
        //获取随机生成的随机码
        $code = $this -> mkCode();
        //通过类的属性指定图形的大小,默认是100,20
        $canvas = imagecreatetruecolor($this -> _width, $this -> _height);
        //随机生成一个颜色的画笔
        $paint = imagecolorallocate($canvas,mt_rand(0,255),mt_rand(0,255),mt_rand(0,255));
        //把背景的颜色进行改变,默认是黑色的。
        imagefill($canvas, 10, 10, $paint);
        //创建一个画随机码的笔,颜色也是随机生成的。
        $paint_str = imagecolorallocate($canvas,mt_rand(0,255),mt_rand(0,255),mt_rand(0,255));
        //把随机码打印在画布上。
        imagestring($canvas, 4, 20, 2, $code, $paint_str);
        //绘制干扰点的颜色
        $paint_pixel = imagecolorallocate($canvas,mt_rand(0,255),mt_rand(0,255),mt_rand(0,255));
        //通过类的属性指定需要多少个干扰点。
        for($i = 0; $i < $this -> _pixel; $i++){
            //绘制不同的干扰点,而绘制的位置也是随机生成的。
            imagesetpixel($canvas, mt_rand(0,imagesx($canvas)),  mt_rand(0,imagesy($canvas)), $paint_pixel);
        }
        header("content-type:image/png");
        imagepng($canvas);
        imagedestroy($canvas);
    }
  1. 先通过本类的方法获取一个随机生成的随机数。
  2. 把画布的背景进行不同颜色的转变
  3. 把随机码绘制在画布上。
  4. 通过循环绘制不同的干扰点。

源代码:

<?php

    Class Cap{
        //生成的字符的长度
        private $_len = 4;
        //干扰点的个数
        private $_pixel = 100;

        private $_width = 100;
        private $_height = 20;

        public function __set($pro,$val){
            if(property_exists($this, $pro)){
                $this -> $pro = $val;
            }
        }

        /*
        生成随机码
         */
        private function mkCode(){
            //通过类的参数获取需要的随机数的个数。这个值可以自由的指定
            $len = $this -> _len;
            //我们生成的随机数的字母喝数字就是在这里面进行随机生成。
            $str = 'ABCDEFGHIGKLMNOPQRST1234567890';
            $code = '';
            //通过循环的生成随机数进行获取
            for($i = 0; $i < $len; $i++){
                //生成随机数
                $j = mt_rand(0,strlen($str)-1);
                //把随机生成的随机数拼接起来。
                $code .= $str[$j];
            }
            //把生成的随机数,保存在session中,便于当我们输入验证码是验证是否正确。
            @session_start();
            $_SESSION['code'] = $code;
            return $code;
        }
        //生成验证码
        public function makeImage(){
            //获取随机生成的随机码
            $code = $this -> mkCode();
            //通过类的属性指定图形的大小,默认是100,20
            $canvas = imagecreatetruecolor($this -> _width, $this -> _height);
            //随机生成一个颜色的画笔
            $paint = imagecolorallocate($canvas,mt_rand(0,255),mt_rand(0,255),mt_rand(0,255));
            //把背景的颜色进行改变,默认是黑色的。
            imagefill($canvas, 10, 10, $paint);
            //创建一个画随机码的笔,颜色也是随机生成的。
            $paint_str = imagecolorallocate($canvas,mt_rand(0,255),mt_rand(0,255),mt_rand(0,255));
            //把随机码打印在画布上。
            imagestring($canvas, 4, 20, 2, $code, $paint_str);
            //绘制干扰点的颜色
            $paint_pixel = imagecolorallocate($canvas,mt_rand(0,255),mt_rand(0,255),mt_rand(0,255));
            //通过类的属性指定需要多少个干扰点。
            for($i = 0; $i < $this -> _pixel; $i++){
                //绘制不同的干扰点,而绘制的位置也是随机生成的。
                imagesetpixel($canvas, mt_rand(0,imagesx($canvas)),  mt_rand(0,imagesy($canvas)), $paint_pixel);
            }
            header("content-type:image/png");
            imagepng($canvas);
            imagedestroy($canvas);
        }

        //判断验证码和填写的验证码是否正确
        public function checkCode($code){
            @session_start();
            if(strtolower($code) === strtolower($_SESSION['code'])){
                return true;
            }
            return false;
        }
    }
    $cap = new Cap();
    $cap -> makeImage();

运行代码,每次刷新都可以生成一个不同的验证码。

总结

PHP的图形绘制,简单的可以理解为不同的函数的利用。上面的案列几乎都是函数的利用。同时也要理解绘制图形的步骤,在绘制图形时,注意坐标是那几个参数。

相关文章