PHP使用imagick扩展来合并图像

xiaoxiao2021-02-28  137

本文通过一个实例对比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" ; }

效果图

转载请注明原文地址: https://www.6miu.com/read-19032.html

最新回复(0)