由于相似图片搜索的php实现的 API 不怎么符合我的用途,所以我重新定义 API 的架构,改写成比较简单的函数方式,虽然还是用对象的方式包装。
<br /><?php <br />/** <br />* 图片相似度比较 <br />* <br />* @version $Id: ImageHash.php 4429 2012-04-17 13:20:31Z jax $ <br />* @author jax.hu <br />* <br />* <code> <br />* //Sample_1 <br />* $aHash = ImageHash::hashImageFile('wsz.11.jpg'); <br />* $bHash = ImageHash::hashImageFile('wsz.12.jpg'); <br />* var_dump(ImageHash::isHashSimilar($aHash, $bHash)); <br />* <br />* //Sample_2 <br />* var_dump(ImageHash::isImageFileSimilar('wsz.11.jpg', 'wsz.12.jpg')); <br />* </code> <br />*/ <br /> <br />class ImageHash { <br /> <br /> /**取样倍率 1~10 <br /> * @access public <br /> * @staticvar int <br /> * */ <br /> public static $rate = 2; <br /> <br /> /**相似度允许值 0~64 <br /> * @access public <br /> * @staticvar int <br /> * */ <br /> public static $similarity = 80; <br /> <br /> /**图片类型对应的开启函数 <br /> * @access private <br /> * @staticvar string <br /> * */ <br /> private static $_createFunc = array( <br /> IMAGETYPE_GIF =>'imageCreateFromGIF', <br /> IMAGETYPE_JPEG =>'imageCreateFromJPEG', <br /> IMAGETYPE_PNG =>'imageCreateFromPNG', <br /> IMAGETYPE_BMP =>'imageCreateFromBMP', <br /> IMAGETYPE_WBMP =>'imageCreateFromWBMP', <br /> IMAGETYPE_XBM =>'imageCreateFromXBM', <br /> ); <br /> <br /> <br /> /**从文件建立图片 <br /> * @param string $filePath 文件地址路径 <br /> * @return resource 当成功开启图片则传递图片 resource ID,失败则是 false <br /> * */ <br /> public static function createImage($filePath){ <br /> if(!file_exists($filePath)){ return false; } <br /> <br /> /*判断文件类型是否可以开启*/ <br /> $type = exif_imagetype($filePath); <br /> if(!array_key_exists($type,self::$_createFunc)){ return false; } <br /> <br /> $func = self::$_createFunc[$type]; <br /> if(!function_exists($func)){ return false; } <br /> <br /> return $func($filePath); <br /> } <br /> <br /> <br /> /**hash 图片 <br /> * @param resource $src 图片 resource ID <br /> * @return string 图片 hash 值,失败则是 false <br /> * */ <br /> public static function hashImage($src){ <br /> if(!$src){ return false; } <br /> <br /> /*缩小图片尺寸*/ <br /> $delta = 8 * self::$rate; <br /> $img = imageCreateTrueColor($delta,$delta); <br /> imageCopyResized($img,$src, 0,0,0,0, $delta,$delta,imagesX($src),imagesY($src)); <br /> <br /> /*计算图片灰阶值*/ <br /> $grayArray = array(); <br /> for ($y=0; $y<$delta; $y++){ <br /> for ($x=0; $x<$delta; $x++){ <br /> $rgb = imagecolorat($img,$x,$y); <br /> $col = imagecolorsforindex($img, $rgb); <br /> $gray = intval(($col['red']+$col['green']+$col['blue'])/3)& 0xFF; <br /> <br /> $grayArray[] = $gray; <br /> } <br /> } <br /> imagedestroy($img); <br /> <br /> /*计算所有像素的灰阶平均值*/ <br /> $average = array_sum($grayArray)/count($grayArray); <br /> <br /> /*计算 hash 值*/ <i style="color:transparent">@本文来源gaodai$ma#com搞$代*码6网</i><b>搞代gaodaima码</b> <br /> $hashStr = ''; <br /> foreach ($grayArray as $gray){ <br /> $hashStr .= ($gray>=$average) ? '1' : '0'; <br /> } <br /> <br /> return $hashStr; <br /> } <br /> <br /> <br /> /**hash 图片文件 <br /> * @param string $filePath 文件地址路径 <br /> * @return string 图片 hash 值,失败则是 false <br /> * */ <br /> public static function hashImageFile($filePath){ <br /> $src = self::createImage($filePath); <br /> $hashStr = self::hashImage($src); <br /> imagedestroy($src); <br /> <br /> return $hashStr; <br /> } <br /> <br /> <br /> /**比较两个 hash 值,是不是相似 <br /> * @param string $aHash A图片的 hash 值 <br /> * @param string $bHash B图片的 hash 值 <br /> * @return bool 当图片相似则传递 true,否则是 false <br /> * */ <br /> public static function isHashSimilar($aHash, $bHash){ <br /> $aL = strlen($aHash); $bL = strlen($bHash); <br /> if ($aL !== $bL){ return false; } <br /> <br /> /*计算容许落差的数量*/ <br /> $allowGap = $aL*(100-self::$similarity)/100; <br /> <br /> /*计算两个 hash 值的汉明距离*/ <br /> $distance = 0; <br /> for($i=0; $i<$aL; $i++){ <br /> if ($aHash{$i} !== $bHash{$i}){ $distance++; } <br /> } <br /> <br /> return ($distance<=$allowGap) ? true : false; <br /> } <br /> <br /> <br /> /**比较两个图片文件,是不是相似 <br /> * @param string $aHash A图片的路径 <br /> * @param string $bHash B图片的路径 <br /> * @return bool 当图片相似则传递 true,否则是 false <br /> * */ <br /> public static function isImageFileSimilar($aPath, $bPath){ <br /> $aHash = ImageHash::hashImageFile($aPath); <br /> $bHash = ImageHash::hashImageFile($bPath); <br /> return ImageHash::isHashSimilar($aHash, $bHash); <br /> } <br /> <br />} <br />