PHP通过哈希算法来实现搜索相似图片

分类:PHP_Python| 发布:佚名| 查看:394 | 发表时间:2016/1/11

感知哈希算法

count < =5 匹配最相似
count > 10 两张不同的图片
var_dump(ImageHash::run(‘./1.png', ‘./camnpr.jpg'));

01<?php
02class ImageHash {
03  const FILE_NOT_FOUND = '-1';
04  const FILE_EXTNAME_ILLEGAL = '-2';
05  private function __construct() {}
06  public static function run($src1, $src2) {
07    static $self;
08    if(!$self) $self = new static;
09    if(!is_file($src1) || !is_file($src2)) exit(self::FILE_NOT_FOUND);
10    $hash1 = $self->getHashValue($src1);
11    $hash2 = $self->getHashValue($src2);
12    if(strlen($hash1) !== strlen($hash2)) return false;
13    $count = 0;
14    $len = strlen($hash1);
15    for($i = 0; $i < $len; $i++) if($hash1[$i] !== $hash2[$i]) $count++;
16    return $count <= 10 ? true : false;
17  }
18  public function getImage($file) {
19    $extname = pathinfo($file, PATHINFO_EXTENSION);
20    if(!in_array($extname, ['jpg','jpeg','png','gif'])) exit(self::FILE_EXTNAME_ILLEGAL);
21    $img = call_user_func('imagecreatefrom'. ( $extname == 'jpg' ? 'jpeg' : $extname ) , $file);
22    return $img;
23  }
24  public function getHashValue($file) {
25    $w = 8;
26    $h = 8;
27    $img = imagecreatetruecolor($w, $h);
28    list($src_w, $src_h) = getimagesize($file);
29    $src = $this->getImage($file);
30    imagecopyresampled($img, $src, 0, 0, 0, 0, $w, $h, $src_w, $src_h);
31    imagedestroy($src);
32    $total = 0;
33    $array = array();
34    for( $y = 0; $y < $h; $y++) {
35      for ($x = 0; $x < $w; $x++) {
36        $gray = (imagecolorat($img, $x, $y) >> 8) & 0xFF;
37        if(!isset($array[$y])) $array[$y] = array();
38        $array[$y][$x] = $gray;
39        $total += $gray;
40      }
41    }
42    imagedestroy($img);
43    $average = intval($total / ($w * $h * 2));
44    $hash = '';
45    for($y = 0; $y < $h; $y++) {
46      for($x = 0; $x < $w; $x++) {
47        $hash .= ($array[$y][$x] >= $average) ? '1' : '0';
48      }
49    }
50    var_dump($hash);
51    return $hash;
52  }
53}
54var_dump(ImageHash::run('./1.png', './camnpr.jpg'));

方法二:

01hash($f);
02 return $isString ? $result[0] : $result;
03 public function checkIsSimilarImg($imgHash, $otherImgHash){
04 if (file_exists($imgHash) && file_exists($otherImgHash)){
05  $imgHash = $this->run($imgHash);
06  $otherImgHash = $this->run($otherImgHash);
07 }
08 if (strlen($imgHash) !== strlen($otherImgHash)) return false;
09 $count = 0;
10 $len = strlen($imgHash);
11 for($i=0;$i<$len;$i++){
12  if ($imgHash{$i} !== $otherImgHash{$i}){
13  $count++;
14  }
15 }
16 return $count <= (5 * $rate * $rate) ? true : false;
17 }
18 public function hash($file){
19 if (!file_exists($file)){
20  return false;
21 }
22 $height = 8 * $this->rate;
23 $width = 8 * $this->rate;
24 $img = imagecreatetruecolor($width, $height);
25 list($w, $h) = getimagesize($file);
26 $source = $this->createImg($file);
27 imagecopyresampled($img, $source, 0, 0, 0, 0, $width, $height, $w, $h);
28 $value = $this->getHashValue($img);
29 imagedestroy($img);
30 return $value;
31 }
32 public function getHashValue($img){
33 $width = imagesx($img);
34 $height = imagesy($img);
35 $total = 0;
36 $array = array();
37 for ($y=0;$y<$height;$y++){
38  for ($x=0;$x<$width;$x++){
39  $gray = ( imagecolorat($img, $x, $y) >> 8 ) & 0xFF;
40  if (!is_array($array[$y])){
41   $array[$y] = array();
42  }
43  $array[$y][$x] = $gray;
44  $total += $gray;
45  }
46 }
47 $average = intval($total / (64 * $this->rate * $this->rate));
48 $result = '';
49 for ($y=0;$y<$height;$y++){
50  for ($x=0;$x<$width;$x++){
51  if ($array[$y][$x] >= $average){
52   $result .= '1';
53  }else{
54   $result .= '0';
55  }
56  }
57 }
58 return $result;
59 }
60 public function createImg($file){
61 $ext = $this->getFileExt($file);
62 if ($ext === 'jpeg') $ext = 'jpg';
63 $img = null;
64 switch ($ext){
65  case 'png' : $img = imagecreatefrompng($file);break;
66  case 'jpg' : $img = imagecreatefromjpeg($file);break;
67  case 'gif' : $img = imagecreatefromgif($file);
68 }
69 return $img;
70 }
71 public function getFileExt($file){
72 $infos = explode('.', $file);
73 $ext = strtolower($infos[count($infos) - 1]);
74 return $ext;
75 }
76}

调用方式如下:

1require_once "Imghash.class.php";
2$instance = ImgHash::getInstance();
3$result = $instance->checkIsSimilarImg('chenyin/IMG_camnpr.png', 'chenyin/IMG_camnpr.JPG');

如果$result值为true, 则表明2个图片相似,否则不相似。

365据说看到好文章不转的人,服务器容易宕机
原创文章如转载,请注明:转载自郑州网建-前端开发 http://camnpr.com/
本文链接:http://camnpr.com/php-python/2224.html