java phash_pHash的Java实现

该博客介绍了如何在Java中实现pHash算法,用于图像特征提取。通过读取图像,进行灰度处理、缩小尺寸、DCT变换、计算平均值及哈希值,从而计算图像的相似度。
摘要由CSDN通过智能技术生成

此算法中的DCT变换是从 http://blog.csdn.net/luoweifu/article/details/8214959抄过来的,因为这种需要大量的复杂的数学运算,我看不来,完全不懂...都已经还给老师了...

package com.imageretrieval.features;

import com.imageretrieval.utils.ImageUtil;

/**

* pHash

* 参考链接:http://blog.csdn.net/zouxy09/article/details/17471401

* http://blog.csdn.net/luoweifu/article/details/8220992

* @author VenyoWang

*

*/

public class PHash {

public static void main(String[] args) {

String hash = getFeatureValue("");

String hash1 = getFeatureValue("");

System.out.println(hash);

System.out.println(hash1);

System.out.println(calculateSimilarity(hash, hash1));

}

public static String getFeatureValue(String imagePath) {

// 缩小尺寸,简化色彩

int[][] grayMatrix = getGrayPixel(imagePath, 32, 32);

// 计算DCT

grayMatrix = DCT(grayMatrix, 32);

// 缩小DCT,计算平均值

int[][] newMatrix = new int[8][8];

double average = 0;

for(int i = 0; i < 8; i++){

for(int j = 0; j < 8; j++){

newMatrix[i][j] = grayMatrix[i][j];

average += grayMatrix[i][j];

}

}

average /= 64.0;

// 计算hash值

String hash = "";

for(int i = 0; i < 8; i++){

for(int j = 0; j < 8; j++){

if(newMatrix[i][j] < average){

hash += '0';

}

else{

hash += '1';

}

}

}

return hash;

}

public static int[][] getGrayPixel(String imagePath, int width, int height) {

BufferedImage bi = null;

try {

bi = resizeImage(imagePath, width, height, BufferedImage.TYPE_INT_RGB);

} catch (Exception e) {

e.printStackTrace();

return null;

}

int minx = bi.getMinX();

int miny = bi.getMinY();

int[][] matrix = new int[width - minx][height - miny];

for (int i = minx; i < width; i++) {

for (int j = miny; j < height; j++) {

int pixel = bi.getRGB(i, j);

int red = (pixel & 0xff0000) >> 16;

int green = (pixel & 0xff00) >> 8;

int blue = (pixel & 0xff);

int gray = (int) (red * 0.3 + green * 0.59 + blue * 0.11);

matrix[i][j] = gray;

}

}

return matrix;

}

public static BufferedImage resizeImage(String srcImgPath, int width, int height, int imageType)

throws IOException {

File srcFile = new File(srcImgPath);

BufferedImage srcImg = ImageIO.read(srcFile);

BufferedImage buffImg = null;

buffImg = new BufferedImage(width, height, imageType);

buffImg.getGraphics().drawImage(srcImg.getScaledInstance(width, height, Image.SCALE_SMOOTH), 0, 0, null);

return buffImg;

}

/**

* 用于计算pHash的相似度

* 相似度为1时,图片最相似

* @param str1

* @param str2

* @return

*/

public static double calculateSimilarity(String str1, String str2) {

int num = 0;

for(int i = 0; i < 64; i++){

if(str1.charAt(i) == str2.charAt(i)){

num++;

}

}

return ((double)num) / 64.0;

}

/**

* 离散余弦变换

* @author luoweifu

*

* @param pix

* 原图像的数据矩阵

* @param n

* 原图像(n*n)的高或宽

* @return 变换后的矩阵数组

*/

public static int[][] DCT(int[][] pix, int n) {

double[][] iMatrix = new double[n][n];

for (int i = 0; i < n; i++) {

for (int j = 0; j < n; j++) {

iMatrix[i][j] = (double) (pix[i][j]);

}

}

double[][] quotient = coefficient(n); // 求系数矩阵

double[][] quotientT = transposingMatrix(quotient, n); // 转置系数矩阵

double[][] temp = new double[n][n];

temp = matrixMultiply(quotient, iMatrix, n);

iMatrix = matrixMultiply(temp, quotientT, n);

for (int i = 0; i < n; i++) {

for (int j = 0; j < n; j++) {

pix[i][j] = (int) (iMatrix[i][j]);

}

}

return pix;

}

/**

* 求离散余弦变换的系数矩阵

* @author luoweifu

*

* @param n

* n*n矩阵的大小

* @return 系数矩阵

*/

private static double[][] coefficient(int n) {

double[][] coeff = new double[n][n];

double sqrt = 1.0 / Math.sqrt(n);

for (int i = 0; i < n; i++) {

coeff[0][i] = sqrt;

}

for (int i = 1; i < n; i++) {

for (int j = 0; j < n; j++) {

coeff[i][j] = Math.sqrt(2.0 / n) * Math.cos(i * Math.PI * (j + 0.5) / (double) n);

}

}

return coeff;

}

/**

* 矩阵转置

* @author luoweifu

*

* @param matrix

* 原矩阵

* @param n

* 矩阵(n*n)的高或宽

* @return 转置后的矩阵

*/

private static double[][] transposingMatrix(double[][] matrix, int n) {

double nMatrix[][] = new double[n][n];

for (int i = 0; i < n; i++) {

for (int j = 0; j < n; j++) {

nMatrix[i][j] = matrix[j][i];

}

}

return nMatrix;

}

/**

* 矩阵相乘

* @author luoweifu

*

* @param A

* 矩阵A

* @param B

* 矩阵B

* @param n

* 矩阵的大小n*n

* @return 结果矩阵

*/

private static double[][] matrixMultiply(double[][] A, double[][] B, int n) {

double nMatrix[][] = new double[n][n];

int t = 0;

for (int i = 0; i < n; i++) {

for (int j = 0; j < n; j++) {

t = 0;

for (int k = 0; k < n; k++) {

t += A[i][k] * B[k][j];

}

nMatrix[i][j] = t;

}

}

return nMatrix;

}

}

Java是一种广泛使用的面向对象的编程语言,由Sun Microsystems公司于1995年5月正式发布。它的设计目标是“一次编写,到处运行(Write Once, Run Anywhere)”,这意味着开发者可以使用Java编写应用程序,并在支持Java的任何平台上无需重新编译即可运行,这得益于其独特的跨平台性,通过Java虚拟机(JVM)实现不同操作系统上的兼容。 Java的特点包括: 面向对象:Java全面支持面向对象的特性,如封装、继承和多态,使得代码更易于维护和扩展。 安全:Java提供了丰富的安全特性,如禁止指针运算、自动内存管理和异常处理机制,以减少程序错误和恶意攻击的可能性。 可移植性:Java字节码可以在所有安装了JVM的设备上执行,从服务器到嵌入式系统,再到移动设备和桌面应用。 健壮性与高性能:Java通过垃圾回收机制确保内存的有效管理,同时也能通过JIT编译器优化来提升运行时性能。 标准库丰富:Java拥有庞大的类库,如Java SE(Java Standard Edition)包含基础API,用于开发通用应用程序;Java EE(Java Enterprise Edition)提供企业级服务,如Web服务、EJB等;而Java ME(Java Micro Edition)则针对小型设备和嵌入式系统。 社区活跃:Java有着全球范围内庞大的开发者社区和开源项目,持续推动技术进步和创新。 多线程支持:Java内建对多线程编程的支持,使并发编程变得更加简单直接。 动态性:Java可以通过反射、注解等机制实现在运行时动态加载类和修改行为,增加了程序的灵活性。 综上所述,Java凭借其强大的特性和广泛的适用范围,在企业级应用、互联网服务、移动开发等领域均扮演着举足轻重的角色,是现代软件开发不可或缺的重要工具之一。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值