PHP使用imagick扩展来合并图像

本文通过一个实例对比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
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值