当前位置:  编程技术>java/j2ee

JAVA hashCode使用方法详解

    来源: 互联网  发布时间:2014-10-29

    本文导语:  一.问题引入谈到hashCode就不得不说equals方法,二者均在Object类里,由于Object类是所有类的基类,所以一切类里都可以重写这两个方法。要想较清晰的理解,需要先知道容器Collection,Set,list,Map(key值不可重复),Set元素无序不重...

一.问题引入
谈到hashCode就不得不说equals方法,二者均在Object类里,由于Object类是所有类的基类,所以一切类里都可以重写这两个方法。
要想较清晰的理解,需要先知道容器Collection,Set,list,Map(key值不可重复),Set元素无序不重复,list元素有序可重复,那么JVM是如何确定不同的元素的呢?
难道是逐个比较么,那样效率就太低了,JVM采用hash的方法(hash地址不一定是实际的物理地址),看看这个地址上是否有内容,没的话就认为不存在相同对象……
 且看下面分解……

二.问题分析
首先equals()和hashcode()这两个方法都是从object类中继承过来的,equals()方法在object类中定义如下:

代码如下:

public boolean equals(Object obj) {
    return (this == obj);
}

从声明看出很明显是对两个对象的地址值进行的比较(即比较引用是否相同)。但是我们必需清楚,当String 、Math、还有Integer、Double。。。。等这些封装类在使用equals()方法时,已经覆盖了object类的
equals()方法。

2. 其次是hashcode() 方法,在object类中定义如下:
public native int hashCode();
说明是一个本地方法,它的实现是根据本地机器相关的。

代码如下:

public int hashCode() {
    int h = hash;
    if (h == 0) {

   nt off = offset;

   char val[] = value;

   int len = count;
 

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


  h = 31*h + val[off++];

   }

   hash = h;
    }
    return h;
}


解释一下这个程序: s[0]*31^(n-1) + s[1]*31^(n-2) + ... + s[n-1] ,可以看出hash地址不一定是实际的内存地址。

 3. 若干规范
若重写equals(Object obj)方法,有必要重写hashcode()方法,确保通过equals(Object obj)方法判断结果为true的两个对象具备相等的hashcode()返回值。说得简单点就是:“如果两个对象相同,那么他们的hashcode应该 相等”。不过请注意:这个只是规范,如果你非要写一个类让equals(Object obj)返回true而hashcode()返回两个不相等的值,编译和运行都是不会报错的。不过这样违反了Java规范,程序也就埋下了BUG。
如果equals(Object obj)返回false,即两个对象“不相同”,并不要求对这两个对象调用hashcode()方法得到两个不相同的数(更印证了hash地址不一定是实际的内存地址)。说的简单点就是:“如果两个对象不相同,他们的hashcode可能相同”。
根据这两个规范,不难得到如下推论:
1、如果两个对象equals,Java运行时环境会认为他们的hashcode一定相等。
2、如果两个对象不equals,他们的hashcode有可能相等。
3、如果两个对象hashcode相等,他们不一定equals(我理解是由于hash冲突造成的)。
4、如果两个对象hashcode不相等,他们一定不equals。

三.问题解决
 测试hashCode和equals方法的使用……

代码如下:

import java.util.HashMap; 
import java.util.Map;  
class A { 

    @Override 
    public boolean equals(Object obj) { 

   System.out.println("判断equals"); 

   return true; 
    } 

    @Override 
    public int hashCode() { 

   System.out.println("判断hashcode"); 

   return 1; 
    } 


 
public class Test { 

    public static void main(String[] args) { 

   Map map = new HashMap(); 

   map.put(new A(), new Object()); 

   map.put(new A(), new Object()); 

 


   System.out.println(map.size()); 
    } 


}


输出:

判断hashcode 
判断hashcode 
判断equals 
2


针对结果分析如下:
可以看出,JRE会调用new A()这个对象的hashcode()方法。其中:打印出的第一行“判断hashcode”是第一次map.put(new A(), new Object())所打印出的。 接下来的“判断hashcode”和“判断equals”是第二次map.put(new A(), new Object())所打印出来的。当第一次map.put(new A(), new Object())的时候,显然,这时候没有相同的,因为这个map中都还没有东西,所以这时候hashcode不相等,则没有必要再调用equals(Object obj)方法了。当第二次map.put(new A(), new Object())的时候,JRE这时候发现了map中有两个相同的hashcode(因为我重写了A类的hashcode()方法永远都返回1),所以有必要调用equals(Object obj)方法进行判断了。然后发现两个对象不equals(因为我重写了equals(Object obj)方法,永远都返回false)。这时候判断结束,判断结果:两次存入的对象不是相同的对象。所以最后打印map的长度的时候显示结果是:2。

四.若干注事事项
我们还应该注意,Java语言对equals()的要求如下,这些要求是必须遵循的:

对称性:如果x.equals(y)返回是“true”,那么y.equals(x)也应该返回是“true”。
反射性:x.equals(x)必须返回是“true”。
传递性:如果x.equals(y)返回是“true”,而且y.equals(z)返回是“true”,那么z.equals(x)也应该返回是“true”。
一致性:如果x.equals(y)返回是“true”,只要x和y内容一直不变,不管你重复x.equals(y)多少次,返回都是“true”。
任何情况下,x.equals(null),永远返回是“false”;x.equals(和x不同类型的对象)永远返回false

 以上这五点是重写equals()方法时,必须遵守的准则,如果违反会出现意想不到的结果,请大家一定要遵守……


    
 
 

您可能感兴趣的文章:

  • JAVA中不赞成使用(Deprecated)的方法是否可以使用
  • java堆栈类使用实例(java中stack的使用方法)
  • 如何将GTK中的CallBack函数封装成类似JAVA接口类的使用方法
  • java用静态工厂代替构造函数使用方法和优缺点
  • Java中的this指针使用方法分享
  • Java输入流Scanner/BufferedReader使用方法示例
  • 谁能告诉我哪里能找到java包内部类及方法使用介绍
  • java.exe和javaw.exe的区别及使用方法
  • Java Swing 非常漂亮外观Nimbus的使用方法实例
  • java使用正则表达为数字添加千位符的简单方法
  • java this super使用方法详解
  • java的split方法使用示例
  • 请问Adapter类在整个Java API中的位置是?以及它的作用和使用方法?
  • java system类使用方法示例 获取系统信息
  • 请教:使用java.net中httpURLConnection类模拟post方法?、、全部分数送上!
  • ?请教:十万火急使用java.net中httpURLConnection类模拟post方法?、、全部分数送
  • Java使用JNI方法调用DLL问题
  • 请问在java中有没有在dos下接受键盘输入的方法,如何使用?谢谢
  • java线程之使用Runnable接口创建线程的方法
  • java中"==" 与equals方法的使用
  • java map(HashMap TreeMap)用法:初始化,遍历和排序详解
  • 哪位java同门师兄有《java2编程详解》电子文档,注意不是影印版
  • 请问哪儿有java2编程详解的电子书下载??本人急需!!跟贴有分!!!
  • 谁有JAVA的类库详解或下载地址?
  • 请问那里有《JAVA2编程详解》可以下载?
  • 《Java 2 编程详解》程序清单14.12中的一个问题。
  • 谁有电子版的《Java编程思想第二版(Thinking in java second)》和《Java2编程详解(special edition java2)》?得到给分
  • 我非常想知道JAVA跟C/C++对于硬件控制的能力孰强孰弱.(菜鸟问题,要详解,在线等待)
  • Java中的随机数详解
  • 深入分析Java内存区域的使用详解
  • Java加载JDBC驱动程序实例详解
  •  
    本站(WWW.)旨在分享和传播互联网科技相关的资讯和技术,将尽最大努力为读者提供更好的信息聚合和浏览方式。
    本站(WWW.)站内文章除注明原创外,均为转载、整理或搜集自网络。欢迎任何形式的转载,转载请注明出处。












  • 相关文章推荐
  • java命名空间java.security.cert类certpath的类成员方法: hashcode定义及介绍
  • java里的hashcode是用来干嘛的
  • java命名空间java.util接口collection<e>的类成员方法: hashcode定义及介绍
  • java中的hashCode方法小例子
  • java命名空间javax.rmi.corba接口stubdelegate的类成员方法: hashcode定义及介绍
  • java中hashCode方法与equals方法的用法总结
  • java命名空间javax.xml.datatype类duration成员方法: hashcode定义参考
  • java命名空间java.text类decimalformat的类成员方法: hashcode定义及介绍
  • java命名空间java.text类decimalformatsymbols的类成员方法: hashcode定义及介绍
  • java命名空间java.text类numberformat的类成员方法: hashcode定义及介绍
  • java命名空间java.text类dateformat的类成员方法: hashcode定义及介绍
  • java命名空间java.net类httpcookie的类成员方法: hashcode定义及介绍
  • java命名空间javax.swing.tree类treepath的类成员方法: hashcode定义及介绍
  • java命名空间java.text类dateformatsymbols的类成员方法: hashcode定义及介绍
  • java命名空间javax.security.auth.kerberos类kerberosprincipal的类成员方法: hashcode定义及介绍
  • java命名空间java.util接口list<e>的类成员方法: hashcode定义及介绍
  • java命名空间java.awt类renderinghints的类成员方法: hashcode定义及介绍
  • java命名空间java.awt.font类numericshaper的类成员方法: hashcode定义及介绍
  • java命名空间java.util类locale的类成员方法: hashcode定义及介绍
  • java命名空间javax.naming类refaddr的类成员方法: hashcode定义及介绍
  • java命名空间javax.sound.sampled类reverbtype的类成员方法: hashcode定义及介绍
  • java命名空间java.sql类types的类成员方法: java_object定义及介绍
  • 我想学JAVA ,是买THINK IN JAVA 还是JAVA2核心技术:卷1 好???
  • java命名空间java.awt.datatransfer类dataflavor的类成员方法: imageflavor定义及介绍
  • 请问Java高手,Java的优势在那里??,Java主要适合于开发哪类应用程序
  • java命名空间java.lang.management类managementfactory的类成员方法: getcompilationmxbean定义及介绍
  • 如何将java.util.Date转化为java.sql.Date?数据库中Date类型对应于java的哪个Date呢
  • java命名空间java.lang.management接口runtimemxbean的类成员方法: getlibrarypath定义及介绍
  • 本人想学java,请问java程序员的待遇如何,和java主要有几个比较强的方向
  • java命名空间java.lang.management接口runtimemxbean的类成员方法: getstarttime定义及介绍
  • 我对JAVA一窍不通,可惜别人却给我一个Java的project,要我做一个安装程序,请问哪里有JAVA INSTALLER下载,而且我要不要安装java的sdk才能完成此项任务?


  • 站内导航:


    特别声明:169IT网站部分信息来自互联网,如果侵犯您的权利,请及时告知,本站将立即删除!

    ©2012-2021,,E-mail:www_#163.com(请将#改为@)

    浙ICP备11055608号-3