网站的登陆页面或者发布文章页面经常要使用验证码来提高安全性,那么php网站要怎么加入验证码呢?接下来我以php轻量级框架codeigniter为例,简单描述一下思路。
首先,codeigniter的captcha验证码类使用起来不是特别好用,我一般用下面这个类来替代:
<?php
class Captcha
{
private $width;
private $height;
private $codeNum;
private $code;
private $im;
function __construct($width=80, $height=30, $codeNum=4){
$this->width = $width;
$this->height = $height;
$this->codeNum = $codeNum;
}
function showImg(){
//创建图片
$this->createImg();
//设置干扰元素
$this->setDisturb();
//设置验证码
$this->setCaptcha();
//输出图片
$this->outputImg();
}
function getCaptcha(){
$this->createCode();
return $this->code;
}
private function createImg(){
$this->im = imagecreatetruecolor($this->width, $this->height);
$bgColor = imagecolorallocate($this->im, 0, 0, 0);
imagefill($this->im, 0, 0, $bgColor);
}
private function setDisturb(){
$area = ($this->width * $this->height) / 20;
$disturbNum = ($area > 250) ? 250 : $area;
//加入点干扰
for ($i = 0; $i < $disturbNum; $i++) {
$color = imagecolorallocate($this->im, rand(0, 255), rand(0, 255), rand(0, 255));
imagesetpixel($this->im, rand(1, $this->width - 2), rand(1, $this->height - 2), $color);
}
//加入弧线
for ($i = 0; $i <= 5; $i++) {
$color = imagecolorallocate($this->im, rand(128, 255), rand(125, 255), rand(100, 255));
imagearc($this->im, rand(0, $this->width), rand(0, $this->height), rand(30, 300), rand(20, 200), 50, 30, $color);
}
}
private function createCode(){
$str = "23456789abcdefghijkmnpqrstuvwxyzABCDEFGHIJKMNPQRSTUVWXYZ";
for ($i = 0; $i < $this->codeNum; $i++) {
$this->code .= $str{rand(0, strlen($str) - 1)};
}
}
private function setCaptcha(){
for ($i = 0; $i < $this->codeNum; $i++) {
$color = imagecolorallocate($this->im, rand(50, 250), rand(100, 250), rand(128, 250));
$size = rand(floor($this->height / 5), floor($this->height / 3));
$x = floor($this->width / $this->codeNum) * $i + 5;
$y = rand(0, $this->height - 20);
imagechar($this->im, $size, $x, $y, $this->code{$i}, $color);
}
}
private function outputImg(){
if (imagetypes() & IMG_JPG) {
header('Content-type:image/jpeg');
imagejpeg($this->im);
} elseif (imagetypes() & IMG_GIF) {
header('Content-type: image/gif');
imagegif($this->im);
} elseif (imagetype() & IMG_PNG) {
header('Content-type: image/png');
imagepng($this->im);
} else {
die("Don't support image type!");
}
}
}
至于CI框架怎么代替原来的类,我给个教程链接:http://codeigniter.org.cn/user_guide/general/creating_libraries.html
那么这个类库怎么用呢?在你的登陆控制器中加入,假设你的登陆控制器是Login_Controller,在里面加入以下代码:
public function get_code(){
$this->load->library('captcha'); //加载这个代替类
$captcha= $this->captcha->getCaptcha(); //生成的验证码值
$this->session->set_userdata('captcha', $captcha); //保存验证码值
$this->captcha->showImg(); //生成验证码图片
}
这段代码的意思是是将生成的验证码值先存到session中,然后生成验证码图片。
那么表单中怎么获取这个已生成的验证码图片呢?看下面代码:
<input type="text" name="captcha" size="4" placeholder="输入验证码">
<img src="/index.php/login/get_code"></img>
<span style="font-size:14px;"><span style="font-size:14px;"><span style="font-size:12px;color:#FF0000;">注意:img的src属性值,不同的控制器名,值就不同,所以仅供参考,请根据你的情况修改。</span></span></span>
那么提交表单后怎么验证验证码呢?假设form的action为login/validate用post方式提交,那么在login控制器加入以下代码:
public function validata()
{
$code1 = strtolower($this->input->post('captcha')); //获取表单中用户输入的验证码值
$code2 = strtolower($this->session->userdata('captcha')); //获取事先保存于session的验证码值
if($code1 != $code2){
//如果不等则验证失败,输出验证失败信息(这里的代码根据你的情况写吧)
}
}
注意:input里面的name属性值"captcha",是跟控制器里的validate函数中$this->input->post('captcha')的"captcha"值要一致。
原文章链接为:http://www.ifixedbug.com/posts/codeigniter-captcha-library
当然这个类我只是举例子在CI框架上的应用,你也可以触类旁通,将这个类整合到你擅长的框架或者使用在原生代码中,如果你不喜欢生成验证码图片的样子,你也可以通过修改类的函数来完成你的需求。