有时候,为了防止一些恶意程序爬取网页脚本,我们会在请求上设置一些门槛,
最常见的就是使用验证码,还有人机验证脚本,比如谷歌的reCaptcha,CloudFlare的5秒盾,这里只介绍验证码,最常用的是图片验证码。
本文涉及到的大部分函数(以image开头)均为GD库所提供。
首先用imagecreatetruecolor创建个黑色画布:
$img = imagecreatetruecolor(100, 40);
100和40分别是长和宽,就是验证码图片的尺寸,
暂时生成三种颜色,这里使用imagecolorallocate:
- 白色做背景色:
$white = imagecolorallocate($img, 0xFF, 0xFF, 0xFF);
- 黑色为文字色:
$black = imagecolorallocate($img, 0x00, 0x00, 0x00);
- 绿色用来干扰:
$green = imagecolorallocate($img, 0x00, 0xFF, 0x00);
$img即前面创建好的图片,后面三个参数分别是颜色的RGB值,可以是普通的十进制数,范围是0~255,这个函数并不会用其他颜色把原图覆盖掉。
现在,用imagefill把图像填充为白色:
imagefill($img, 0, 0, $white);
然后生成四位随机数字:
$code = '';
for($i = 0; $i < 4; ++$i)
$code .= rand(0, 9);
rand是一个生成随机数的函数,0和9即生成的随机数范围。
不过rand生成的数随机性不好,所以这里改用mt_rand,使用方法一致。
然后将文本显示到图片上:
imagestring($img, 5, 35, 10, $code, $black);
5为字体,35和10为输出坐标,$code就是要显示的文本,$black为文本颜色,这里用前面生成的黑色。
接着,我们给图片加一些干扰色:
// 循环50次是为了提高干扰程度
for($i = 0; $i < 50; ++$i) {
imagesetpixel($img, mt_rand(0,100), mt_rand(0, 40), $black);
imagesetpixel($img, mt_rand(0,100), mt_rand(0, 40), $green);
}
大致解释:用imagesetpixel给$img对应的图像的(mt_rand(0,100), mt_rand(0, 40))加上$black对应的颜色。
最后,我们用imagepng直接输出图片数据,但因为识别问题,应该先用header告诉浏览器图片类型,不然会被当做文本,导致显示成乱码。
// content-type是内容类型
header('content-type: image/png');
imagepng($img);
为了节省资源,最好在不用的时候就释放资源。
imagedestroy($img);
说明:
输出时不一定要用png,也可以是jpeg、gif之类的图片格式,修改对应函数即可。
参考资料:
[1] 图像生成和处理 | PHP手册
1256

被折叠的 条评论
为什么被折叠?



