PHP使用imagick扩展来合并图像

来源:互联网 发布:好奇日报 知乎 编辑:程序博客网 时间:2024/05/23 00:06

本文通过一个实例对比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扩展。下面上代码,有详细注释:

第一步:

  1. // step1: 调整尺寸到 590 x 590
  2. $a = ROOT . '/' . 'a';
  3. // 扫描目录
  4. $dirA = scandir($a);
  5. $im = new Imagick;
  6. foreach ($dirA as $item) {
  7.  
  8. // 跳过目录和缩略图
  9. if ($item === '.' || $item === '..' || strstr($item, '.db')) {
  10. continue;
  11. }
  12.  
  13. // 读取图片
  14. $im->readImage($a . '/' . $item);
  15.  
  16. // 获取图片宽x高
  17. $geo = $im->getImageGeometry();
  18. if ($geo['width'] === 590 && $geo['height'] === 590) {
  19. // 宽高符合,跳过
  20. } else {
  21.  
  22. // 调整尺寸到590 x 590
  23. im->resizeImage(590, 590, Gmagick::FILTER_UNDEFINED, 1, TRUE);
  24. }
  25.  
  26. // 将图片保存到另一目录
  27. $im->writeImage(ROOT . '/_a/' . $item);
  28.  
  29. // 释放资源
  30. $im->destroy();
  31. }

第二步:

  1. // step2: 合并图片和名字
  2. // 扫描目录
  3. $files = scandir(ROOT . '/_a');
  4. $k = 0;
  5. foreach ($files as $item) {
  6. // 跳过目录和缩略图
  7. if ($item === '.' || $item === '..' || strstr($item, '.db')) {
  8. continue;
  9. }
  10. // 文本图片的宽
  11. $twidth = 570;
  12. // 文本图片的高
  13. $theight = 141;
  14. // 获取图片名
  15. $pathinfo = pathinfo($item);
  16. $filename = $pathinfo['filename'];
  17.  
  18. // 初始化图片对象
  19. $text = new Imagick;
  20. // 初始化绘制对象
  21. $draw = new ImagickDraw;
  22. // 设置字体,这里是放到网站的font下的微软雅黑
  23. $draw->setFont('font/msyh.ttf');
  24. // 文字大小
  25. $draw->setFontSize(40);
  26. // 文字颜色
  27. $draw->setFillColor(new ImagickPixel('#000000'));
  28. // 文字对齐方式
  29. $draw->setTextAlignment(Imagick::ALIGN_LEFT);
  30. // 获取文字信息,主要是长宽,因为要实现在图片居中
  31. $a = $text->queryFontMetrics($draw, $filename);
  32. // 添加文字
  33. $draw->annotation(($twidth - $a['textWidth']) / 2, 80, $filename);
  34. // 建立图像
  35. $text->newImage($twidth, $theight, new ImagickPixel('#ffffff'));
  36. // 图片格式
  37. $text->setImageFormat('png');
  38. // 绘制图片
  39. $text->drawImage($draw);
  40.  
  41. // 新建一个空白图片用来做画布
  42. $canvas = new Imagick;
  43. $canvas->newimage(570, 661, 'white');
  44. $canvas->setImageFormat('png');
  45.  
  46. // 读取图片
  47. $pic = new Imagick;
  48. $pic->readImage(ROOT . '/_a/' . $item);
  49. $pic->scaleimage(470, 470, TRUE);
  50.  
  51. // 将图片合并到画布
  52. $canvas->compositeImage($pic, Imagick::COMPOSITE_OVER, 50, 50);
  53. // 将文字合并到画布
  54. $canvas->compositeimage($text, Imagick::COMPOSITE_OVER, 0, 520);
  55. // 保存图片到另一目录
  56. $canvas->writeimage(ROOT . '/com_a/' . $item);
  57. $k++;
  58. echo "{$k} files proceeded.\n";
  59. }

效果图

第三步:

  1. // step3: 合并每20张到一页
  2. // 扫描目录
  3. $files = scandir(ROOT . '/com_a');
  4.  
  5. // 给图片分组
  6. $i = $j = 0;
  7. $group = array();
  8. foreach ($files as $item) {
  9. if ($item === '.' || $item === '..' || strstr($item, '.db')) {
  10. continue;
  11. }
  12. $i++;
  13. $group[$j][] = $item;
  14. if ($i % 20 === 0) {
  15. $j++;
  16. }
  17. }
  18.  
  19. $total = count($group);
  20. // 按组拼接图片,A4纸尺寸,4x5的组合方式
  21. foreach ($group as $k => $v) {
  22. $canvas = new Imagick;
  23. $canvas->newimage(2480, 3508, 'white');
  24. $canvas->setimageformat('png');
  25. $i = $j = 0;
  26. foreach ($v as $item) {
  27. $im = new Imagick(ROOT . '/com_a/' . $item);
  28. // 预留了150的左边距
  29. $x = 150 + $i * 570;
  30. // 130的顶边距
  31. $y = 130 + $j * 661;
  32. $canvas->compositeimage($im, Imagick::COMPOSITE_OVER, $x, $y);
  33. // 每4张一行
  34. if (($i + 1) % 4 === 0) {
  35. $i = 0;
  36. $j++;
  37. } else {
  38. $i++;
  39. }
  40. }
  41. $canvas->writeimage(ROOT . '/merge_a/' . $k . '.png');
  42. $c = $k + 1;
  43. echo "Group {$c}/{$total} done.\n";
  44. }

效果图

0 0
原创粉丝点击