PHP 图片相似度

PHP 图片相似度查询

图片查询数据

<?php
/**
 * 图片相似度比较

 * //Sample_1
 * $aHash = ImageHash::hashImageFile('wsz.11.jpg');
 * $bHash = ImageHash::hashImageFile('wsz.12.jpg');
 * var_dump(ImageHash::isHashSimilar($aHash, $bHash));

 * //Sample_2
 * var_dump(ImageHash::isImageFileSimilar('wsz.11.jpg', 'wsz.12.jpg'));
 */
class ImageHash {
    /**取样倍率 1~10  数值越高匹配越准确
     * @access public
     * @staticvar int
     * */
    public static $rate = 2;
    /**相似度允许值 0~64 数值越高 匹配精度越准确
     * @access public
     * @staticvar int
     * */
    public static $similarity = 89;
    /**图片类型对应的开启函数
     * @access private
     * @staticvar string
     * */
    private static $_createFunc = array(
        IMAGETYPE_GIF =>'imageCreateFromGIF',
        IMAGETYPE_JPEG =>'imageCreateFromJPEG',
        IMAGETYPE_PNG =>'imageCreateFromPNG',
        IMAGETYPE_BMP =>'imageCreateFromBMP',
        IMAGETYPE_WBMP =>'imageCreateFromWBMP',
        IMAGETYPE_XBM =>'imageCreateFromXBM',
    );
    /**比较两个图片文件,是不是相似
     * @param string $aHash A图片的路径
     * @param string $bHash B图片的路径
     * @return bool 当图片相似则传递 true,否则是 false
     * */
    public static function isImageFileSimilar($aPath, $bPath){
        $aHash = ImageHash::hashImageFile($aPath);
        $bHash = ImageHash::hashImageFile($bPath);
        return ImageHash::isHashSimilar($aHash, $bHash);
    }
    /**hash 图片文件
     * @param string $filePath 文件地址路径
     * @return string 图片 hash 值,失败则是 false
     * */
    public static function hashImageFile($filePath){
        $src = self::createImage($filePath);
        $hashStr = self::hashImage($src);
        imagedestroy($src);
        return $hashStr;
    }
    /**从文件建立图片
     * @param string $filePath 文件地址路径
     * @return resource 当成功开启图片则传递图片 resource ID,失败则是 false
     * */
    public static function createImage($filePath){
        if(!file_exists($filePath)){ return false; }
        /*判断文件类型是否可以开启*/
        $type = exif_imagetype($filePath);
        if(!array_key_exists($type,self::$_createFunc)){ return false; }
        $func = self::$_createFunc[$type];
        if(!function_exists($func)){ return false; }
        return $func($filePath);
    }
    /**hash 图片
     * @param resource $src 图片 resource ID
     * @return string 图片 hash 值,失败则是 false
     * */
    public static function hashImage($src){
        if(!$src){ return false; }
        /*缩小图片尺寸*/
        $delta = 8 * self::$rate;
        $img = imageCreateTrueColor($delta,$delta);
        imageCopyResized($img,$src, 0,0,0,0, $delta,$delta,imagesX($src),imagesY($src));
        /*计算图片灰阶值*/
        $grayArray = array();
        for ($y=0; $y<$delta; $y++){
            for ($x=0; $x<$delta; $x++){
                $rgb = imagecolorat($img,$x,$y);
                $col = imagecolorsforindex($img, $rgb);
                $gray = intval(($col['red']+$col['green']+$col['blue'])/3)& 0xFF;
                $grayArray[] = $gray;
            }
        }
        imagedestroy($img);
        /*计算所有像素的灰阶平均值*/
        $average = array_sum($grayArray)/count($grayArray);
        /*计算 hash 值*/
        $hashStr = '';
        foreach ($grayArray as $gray){
            $hashStr .= ($gray>=$average) ? '1' : '0';
        }
        return $hashStr;
    }
    /**比较两个 hash 值,是不是相似
     * @param string $aHash A图片的 hash 值
     * @param string $bHash B图片的 hash 值
     * @return bool 当图片相似则传递 true,否则是 false
     * */
    public static function isHashSimilar($aHash, $bHash){
        $aL = strlen($aHash); $bL = strlen($bHash);
        if ($aL !== $bL){ return false; }
        /*计算容许落差的数量*/
        $allowGap = $aL*(100-self::$similarity)/100;
        /*计算两个 hash 值的汉明距离*/
        $distance = 0;
//        $distance = similar_text($aHash,$bHash);
//        return (($distance-$allowGap) >= $allowGap) ? true : false;
//        dd($allowGap,$distance);
//        if(similar_text($aHash,$bHash) >200){
//            return ($distance<=$allowGap) ? true : false;
//        }
        for($i=0; $i<$aL; $i++){
//            dd($aHash[$i],$bHash[$i],$distance);
            if ($aHash{$i} !== $bHash{$i}){ $distance++; }
        }
        return ($distance<=$allowGap) ? true : false;
    }
}


  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值