最近做一个生成二维码的功能,最初的做法是使用phpqrcode生成png格式的二维码,保存到本地后,使用GD库从这个文件创建图像,再进行添加文字、水印等处理,最后删除本地二维码图片文件,保存处理后的图片。然而,当要生成图片的数目较多时,运行十分缓慢。本文主要讲述尝试代码优化的过程,也只讨论关键部分的代码,其余忽略。提及的知识如下:
-
PHP QR Code
是一个开放源代码的php生成二维码的类库,官网上有下载地址以及实例演示的demo:http://phpqrcode.sourceforge.net
有一点需要注意,PHP环境必须开启支持GD2,打开phpinfo查看是否有如下图显示的内容,有则已开启:
-
GD库
是php处理图形的扩展库,GD库提供了一系列用来处理图片的API,使用GD库可以处理图片,或者生成图片。(是否开启和上图相同)GD库的使用详情可上官网查看:http://php.net/manual/en/book.image.php
优化过程
思路:将二维码保存到本地这个过程去掉,直接生成图像,并在之后的处理中使用。这样在生成二维码数目较多时,运行速度应会有所提高。
首先,生成二维码的代码为:
//引入类库
include 'phpqrcode.php';
// $text所要生成的链接或文字
// $outfile保存二维码的路径,(默认为false,直接在浏览器显示二维码)
// $level纠错级别
// $size点的大小
// $margin为边缘厚度
// $saveandprint表示是否保存二维码并显示
QRcode::png($text, $outfile, $level, $size, $margin, $saveandprint=false);
为了达到预期目标,如果执行QRcode::png()
后,返回的是一个resource类型的图像,那么目的便达成了。首先看一下源码里是怎么写的,打开phpqrcode.php这个文件,找到QRcode类的png方法:
public static function png($text, $outfile = false, $level = QR_ECLEVEL_L, $size = 3, $margin = 4, $saveandprint=false, $back_color = 0xFFFFFF, $fore_color = 0x000000)
{
$enc = QRencode::factory($level, $size, $margin, $back_color, $fore_color);
return $enc->encodePNG($text, $outfile, $saveandprint=false);
}
在这里,先是使用了QRencode类的factory方法,再调用encodePNG方法:
public static function factory($level = QR_ECLEVEL_L, $size = 3, $margin = 4, $back_color = 0xFFFFFF, $fore_color = 0x000000, $cmyk = false)
{
$enc = new QRencode();
/*此处代码省略,详情看源码*/
return $enc;
}
发现$enc其实就是QRencode类的实例化,那么需要找到QRencode类的encodePNG方法:
public function encodePNG($intext, $outfile = false,$saveandprint=false)
{
try {
ob_start();
$tab = $this->encode($intext);
$err = ob_get_contents();
ob_end_clean();
if ($err != '')
QRtools::log($outfile, $err);
$maxSize = (int)(QR_PNG_MAXIMUM_SIZE / (count($tab)+2*$this->margin));
QRimage::png($tab, $outfile, min(max(1, $this->size), $maxSize), $this->margin,$saveandprint, $this->back_color, $this->fore_color);
} catch (Exception $e) {
QRtools::log($outfile, $e->getMessage());
}
}
关键的代码便是QRimage::png()
这里,源码如下:
public static function png($frame, $filename = false, $pixelPerPoint = 4, $outerFrame = 4,$saveandprint=FALSE, $back_color, $fore_color)
{
$image = self::image($frame, $pixelPerPoint, $outerFrame, $back_color, $fore_color);
if ($filename === false) {
Header("Content-type: image/png");
ImagePng($image);
} else {
if($saveandprint===TRUE){
ImagePng($image, $filename);
header("Content-type: image/png");
ImagePng($image);
}else{
ImagePng($image, $filename);
}
}
ImageDestroy($image);
}
private static function image($frame, $pixelPerPoint = 4, $outerFrame = 4, $back_color = 0xFFFFFF, $fore_color = 0x000000)
{
/*此处代码省略,详情看源码*/
return $target_image;
}
可以发现QRimage类的png方法(注意此处与一开始的png方法是在不同的类里)其实已经创建了二维码的图像,只是没有返回给最开始调用的函数,而是将图像保存为文件,或者在浏览器显示。那么只要不让它保存为文件或显示在浏览器中,而是返回,那么目的便达到了。因此对png方法方法的修改如下:
public static function png($frame, $filename = false, $pixelPerPoint = 4, $outerFrame = 4,$saveandprint=FALSE, $back_color, $fore_color)
{
$image = self::image($frame, $pixelPerPoint, $outerFrame, $back_color, $fore_color);
return $image;
}
修改后,QRimage类的png方法便将$image返回给encodePNG方法,需要再将$image返回给QRcode类的png方法,对encodePNG方法的部分代码修改如下:
$image = QRimage::png($tab, $outfile, min(max(1, $this->size), $maxSize), $this->margin,$saveandprint, $this->back_color, $this->fore_color);
return $image;
最后,我们便直调用QRcode::png()
并赋值给变量,不用保存为文件就可以进行添加文字、水印等处理拉!