本文通过一个实例对比Imagick和Gmagick的像素迭代功能:
像素数据生成代码
<?php
$data = array();
for ($row = 0; $row < 100; $row++) {
for ($column = 0; $column < 100; $column++) {
$data[$row][$column] = '#' . str_repeat($column % 10, 6);
}
}
?>
Imagick迭代写像素
<?php
require 'data.php';
$image = new Imagick();
$image->newimage(100, 100, 'white', 'png');
$iterator = $image->getPixelIterator();
foreach ($iterator as $row => $pixels) {
foreach ($pixels as $column => $pixel) {
$pixel->setColor($data[$row][$column]);
}
$iterator->syncIterator();
}
$image->writeimage('pixel.png');
?>
注:在Imagick中利用PixelIterator写像素时,需要调用syncIterator操作(读像素不用)。
Gmagick迭代写像素
<?php
require 'data.php';
$image = new Gmagick();
$image->newimage(100, 100, 'white', 'png');
$pixel = new GmagickPixel();
$draw = new GmagickDraw();
for ($row = 0; $row < 100; $row++) {
for ($column = 0; $column < 100; $column++) {
$pixel->setcolor($data[$row][$column]);
$draw->setfillcolor($pixel);
$draw->point($column, $row);
$image->drawimage($draw);
$pixel->clear();
$draw->clear();
}
}
$image->writeimage('pixel.png');
?>
<?php class ImgCompare { private static $_instance = null; public static $rate = 1; public static function init() { if (self::$_instance === null) { self::$_instance = new self(); } return self::$_instance; } public function doCompare($file) { if(!function_exists('imagecreatetruecolor')) { throw new Exception('GD Library must be load if you want to use the class ImgCompare'); } $is_string = false; if(is_string($file)) { $file = array($file); $is_string = true; } $result = array(); foreach ($file as $f) { $result[] = $this->hash($f); } return $is_string ? $result[0] : $result; } public function checkIsSimilar($img_hash_1,$img_hash_2) { if (file_exists($img_hash_1) && file_exists($img_hash_2)) { $img_hash_1 = self::doCompare($img_hash_1); $img_hash_2 = self::doCompare($img_hash_2); } if(strlen($img_hash_1) !== strlen($img_hash_2)) { return "图片错误"; } $count = 0; $len = strlen($img_hash_1); for ($i=0;$i<$len;$i++) { if($img_hash_1{$i} !== $img_hash_2{$i}) { // 计算 有多少位是不一样的 $count ++; } } // 得到指纹以后,就可以对比不同的图片,看看64位中有多少位是不一样的。在理论上,这等同于计算"汉明距离"(Hamming distance)。 // 如果不相同的数据位不超过5*误差,就说明两张图片很相似;如果大于10*误差,就说明这是两张不同的图片。 return $count; } public function hash($file) { if (!file_exists($file)) { return "图片不存在"; } $height = 8*self::$rate; $width = 8*self::$rate; $img = imagecreatetruecolor($width, $height); list($w,$h) = getimagesize($file); $source = self::createImg($file); // 重采样拷贝部分图像并调整大小 // 将一幅图像中的一块正方形区域拷贝到另一个图像中,平滑地插入像素值,因此,尤其是,减小了图像的大小而仍然保持了极大的清晰度 // 如果源和目标的宽度和高度不同,则会进行相应的图像收缩和拉伸。坐标指的是左上角 // 本函数可用来在同一幅图内部拷贝(如果 dst_image 和 src_image 相同的话)区域,但如果区域交迭的话则结果不可预知。 imagecopyresampled($img, $source, 0, 0, 0, 0, $width, $height, $w, $h); $value = self::getHashValue($img); imagedestroy($img); return $value; } public function getHashValue($img) { $width = imagesx($img); $height = imagesy($img); $total = 0; $array = array(); // 将缩小后的图片,转为64级灰度。也就是说,所有像素点总共只有64种颜色。 for ($y =0;$y<$height;$y++) { for ($x=0;$x<$width;$x++) { // 获取 指定的图形中指定坐标像素的颜色索引值 // 将缩小的图像转为64级灰度 $gray = ( imagecolorat($img, $x, $y) >> 8 ) & 0xFF; if (!is_array($array[$y])) { $array[$y] = array(); } $array[$y][$x] = $gray; $total += $gray; } } // 获取灰度平均值 $average = intval($total/(64*self::$rate*self::$rate)); $result = ''; for ($y=0;$y<$height;$y++) { for ($x=0;$x<$width;$x++) { // 将每个像素的灰度,与平均值进行比较。大于或等于平均值,记为1;小于平均值,记为0 if ($array[$y][$x] >= $average) { $result .= '1'; } else { $result .= '0'; } } } return $result; } public function createImg($file) { $ext = self::getFileExt($file); if ($ext === 'jpeg') $ext = 'jpg'; $img = null; switch ($ext){ case 'png' : $img = imagecreatefrompng($file);break; case 'jpg' : $img = imagecreatefromjpeg($file);break; case 'gif' : $img = imagecreatefromgif($file);break; default:break; } return $img; } public function getFileExt($file){ $infos = explode('.', $file); $ext = strtolower($infos[count($infos) - 1]); if (!in_array($ext,array('jpg','jpeg','png','gif'))) { throw new Exception("file extension must in 'jpg','jpeg','png','gif' "); exit; } return $ext; } } $instance = ImgCompare::init(); $reuslt=$instance->checkIsSimilar('imagetest/5.jpg', 'imagetest/4.jpg'); echo $reuslt;//根据返回的值0为相同 值越大图片差异越大
好俺还会点PHP,好吧,写个小程序来完成拼图。因为图片都是按编号排列的,要求给每个图片都加上编号,于是我的思路是:1.先把所有图片缩放到统一尺寸 2.把每张图片和编号组合到一张图 3.把每20张图再组合到一张图。图片处理用到了ImageMagick和php的imagick扩展。下面上代码,有详细注释:
第一步:
// step1: 调整尺寸到 590 x 590 $a = ROOT . '/' . 'a' ; // 扫描目录 $dirA = scandir ( $a ) ; $im = new Imagick ; foreach ( $dirA as $item ) { // 跳过目录和缩略图 if ( $item === '.' || $item === '..' || strstr ( $item , '.db' ) ) { continue ; } // 读取图片 $im -> readImage ( $a . '/' . $item ) ; // 获取图片宽x高 $geo = $im -> getImageGeometry ( ) ; if ( $geo [ 'width' ] === 590 && $geo [ 'height' ] === 590 ) { // 宽高符合,跳过 } else { // 调整尺寸到590 x 590 im -> resizeImage ( 590 , 590 , Gmagick :: FILTER_UNDEFINED , 1 , TRUE ) ; } // 将图片保存到另一目录 $im -> writeImage (ROOT . '/_a/' . $item ) ; // 释放资源 $im -> destroy ( ) ; } 第二步: // step2: 合并图片和名字 // 扫描目录 $files = scandir (ROOT . '/_a' ) ; $k = 0 ; foreach ( $files as $item ) { // 跳过目录和缩略图 if ( $item === '.' || $item === '..' || strstr ( $item , '.db' ) ) { continue ; } // 文本图片的宽 $twidth = 570 ; // 文本图片的高 $theight = 141 ; // 获取图片名 $pathinfo = pathinfo ( $item ) ; $filename = $pathinfo [ 'filename' ] ; // 初始化图片对象 $text = new Imagick ; // 初始化绘制对象 $draw = new ImagickDraw ; // 设置字体,这里是放到网站的font下的微软雅黑 $draw -> setFont ( 'font/msyh.ttf' ) ; // 文字大小 $draw -> setFontSize ( 40 ) ; // 文字颜色 $draw -> setFillColor ( new ImagickPixel ( '#000000' ) ) ; // 文字对齐方式 $draw -> setTextAlignment (Imagick :: ALIGN_LEFT ) ; // 获取文字信息,主要是长宽,因为要实现在图片居中 $a = $text -> queryFontMetrics ( $draw , $filename ) ; // 添加文字 $draw -> annotation ( ( $twidth - $a [ 'textWidth' ] ) / 2 , 80 , $filename ) ; // 建立图像 $text -> newImage ( $twidth , $theight , new ImagickPixel ( '#ffffff' ) ) ; // 图片格式 $text -> setImageFormat ( 'png' ) ; // 绘制图片 $text -> drawImage ( $draw ) ; // 新建一个空白图片用来做画布 $canvas = new Imagick ; $canvas -> newimage ( 570 , 661 , 'white' ) ; $canvas -> setImageFormat ( 'png' ) ; // 读取图片 $pic = new Imagick ; $pic -> readImage (ROOT . '/_a/' . $item ) ; $pic -> scaleimage ( 470 , 470 , TRUE ) ; // 将图片合并到画布 $canvas -> compositeImage ( $pic , Imagick :: COMPOSITE_OVER , 50 , 50 ) ; // 将文字合并到画布 $canvas -> compositeimage ( $text , Imagick :: COMPOSITE_OVER , 0 , 520 ) ; // 保存图片到另一目录 $canvas -> writeimage (ROOT . '/com_a/' . $item ) ; $k ++; echo "{$k} files proceeded.\n" ; }效果图
第三步:
// step3: 合并每20张到一页 // 扫描目录 $files = scandir (ROOT . '/com_a' ) ; // 给图片分组 $i = $j = 0 ; $group = array ( ) ; foreach ( $files as $item ) { if ( $item === '.' || $item === '..' || strstr ( $item , '.db' ) ) { continue ; } $i ++; $group [ $j ] [ ] = $item ; if ( $i % 20 === 0 ) { $j ++; } } $total = count ( $group ) ; // 按组拼接图片,A4纸尺寸,4x5的组合方式 foreach ( $group as $k => $v ) { $canvas = new Imagick ; $canvas -> newimage ( 2480 , 3508 , 'white' ) ; $canvas -> setimageformat ( 'png' ) ; $i = $j = 0 ; foreach ( $v as $item ) { $im = new Imagick (ROOT . '/com_a/' . $item ) ; // 预留了150的左边距 $x = 150 + $i * 570 ; // 130的顶边距 $y = 130 + $j * 661 ; $canvas -> compositeimage ( $im , Imagick :: COMPOSITE_OVER , $x , $y ) ; // 每4张一行 if ( ( $i + 1 ) % 4 === 0 ) { $i = 0 ; $j ++; } else { $i ++; } } $canvas -> writeimage (ROOT . '/merge_a/' . $k . '.png' ) ; $c = $k + 1 ; echo "Group {$c}/{$total} done.\n" ; }效果图