日期:2014-05-17  浏览次数:20508 次

JAVA中的原理移植到PHP中
一下是我在CSDN上下载的一段JAVA的源代码。目的是计算出图片指纹,
public static String produceFingerPrint(String filename) {
BufferedImage source = ImageHelper.readPNGImage(filename);// 读取文件

int width = 8;
int height = 8;

// 第一步,缩小尺寸。
// 将图片缩小到8x8的尺寸,总共64个像素。这一步的作用是去除图片的细节,只保留结构、明暗等基本信息,摒弃不同尺寸、比例带来的图片差异。
BufferedImage thumb = ImageHelper.thumb(source, width, height, false);

// 第二步,简化色彩。
// 将缩小后的图片,转为64级灰度。也就是说,所有像素点总共只有64种颜色。
int[] pixels = new int[width * height];
for (int i = 0; i < width; i++) {
for (int j = 0; j < height; j++) {
pixels[i * height + j] = ImageHelper.rgbToGray(thumb.getRGB(i, j));
}
}

// 第三步,计算平均值。
// 计算所有64个像素的灰度平均值。
int avgPixel = ImageHelper.average(pixels);

// 第四步,比较像素的灰度。
// 将每个像素的灰度,与平均值进行比较。大于或等于平均值,记为1;小于平均值,记为0。
int[] comps = new int[width * height];
for (int i = 0; i < comps.length; i++) {
if (pixels[i] >= avgPixel) {
comps[i] = 1;
} else {
comps[i] = 0;
}
}

// 第五步,计算哈希值。
// 将上一步的比较结果,组合在一起,就构成了一个64位的整数,这就是这张图片的指纹。组合的次序并不重要,只要保证所有图片都采用同样次序就行了。
StringBuffer hashCode = new StringBuffer();
for (int i = 0; i < comps.length; i+= 4) {
int result = comps[i] * (int) Math.pow(2, 3) + comps[i + 1] * (int) Math.pow(2, 2) + comps[i + 2] * (int) Math.pow(2, 1) + comps[i + 2];
hashCode.append(binaryToHex(result));
}

// 得到指纹以后,就可以对比不同的图片,看看64位中有多少位是不一样的。
return hashCode.toString();
}


现在我要用PHP做网页也是实现这种功能,问哪位高手能给出转换到PHP的源代码呢?
------解决方案--------------------
算法描述的很清楚,实现起来就不难
需要注意的是 php 不支持 64位 的整型数,所以函数提供了几种可能的返回样式
/**
 * 生成指纹
 * $filename 图片文件名,可以是URL。只能是GD支持的图片类型
 * $retmode 返回格式:0 二进制表示 1 数组 2 十六进制表示
 **/
function produceFingerPrint($filename, $retmode=0) {
  $sim = imagecreatefromstring( file_get_contents($filename) );
  $dim = imagecreate(8, 8);
  imagecopyresized($dim, $sim, 0, 0, 0, 0, 8, 8, imagesx($sim), imagesy($sim));
  imagetruecolortopalette($sim, true, 64);
  //imagefilter($sim, IMG_FILTER_GRAYSCALE);

  for($x=0; $x<8; $x++) for($y=0; $y<8; $y++) {
    $c = imagecolorat($dim, $x, $y);
    $p[] = array_sum( imagecolorsforindex($dim, $c))/3;
  }
  $avl = array_sum($p)/count($p);
  $r = '';
  foreach($p as $v) $r .= $v>=$avl ? 1 : 0;

  if($retmode == 0) return $r;
  $p = array_map('bindec', str_split($r, 8));
  if($r