java中引用地址和hashcode值的关系

hashCode和内存地址

hashCode和内存地址关系

首先hashCode和java内存地址是没有直接的关系的,当我们没有重写hashCode方法的时候默认的哈希值是内存地址是永远不相同的。

 Student s=new Student();
	Student a=new Student();
	System.out.println(s.hashCode());//2018699554
	System.out.println(a.hashCode());//1311053135

重写hashCode的后,hashCode值是根据实例对象(对象内容)生成的

Student s=new Student();
	Student a=new Student();
	System.out.println(s.hashCode());//29791
	System.out.println(a.hashCode());//29791

哈希值是根据对应属性的内存地址来生成的。

**注意:**如果我们在对象里重写了Object类的equals方法必定也要重写hashCode方法

//重写的hashCode
public int hashCode() {
		final int prime = 31;
		int result = 1;
		result = prime * result + age;
		result = prime * result + ((name == null) ? 0 : name.hashCode());
		result = prime * result + ((sex == null) ? 0 : sex.hashCode());
		return result;
	}

//系统自带的hashCode(内存地址)
	 public int hashCode() {
        int h = hash;
        if (h == 0 && value.length > 0) {
            char val[] = value;
            for (int i = 0; i < value.length; i++) {
                h = 31 * h + val[i];
            }
            hash = h;
        }
        return h;
    }

hashCode去重复原理

哈希值是为了去重复用的
判断对象是否重复也是根据哈希值来进行判断的而不是通过内存地址
而哈希值的生成又和对象属性具体的值有关
所以生成一个对象就会生成一个哈希值(哈希值不是唯一的是根据属性的值来生成的)

java中内存地址生成策略

Java中的内存地址生成策略包括以下几个步骤:

  1. JVM先为每个Java线程分配一块私有的栈空间,用于存储该线程执行时所需要的局部变量、方法参数、操作数栈、返回值等信息。
  2. 当程序执行到创建新的对象实例的代码时,JVM会在堆内存中为其分配一块连续的内存空间。这个内存空间的大小取决于对象的类型和实例变量的数量。
  3. JVM会分配一个句柄或指针,用来存储对象的引用地址。这个句柄或指针的大小固定为4字节或8字节,与运行的操作系统和JVM的位数有关。
  4. JVM将引用地址存储到栈或操作数栈中。在方法调用结束后,局部变量和方法参数被弹出栈空间,但对象的引用地址仍然保留在栈或操作数栈中,直到该对象不再被引用为止。

需要注意的是,Java中的对象是通过引用来传递和使用的,在方法调用时也是传递引用而非对象本身。因此,每个对象的内存地址都是独立的,它们之间没有任何关联。

System.identityHashCode()获取引用地址

 /**
 String str="i"与 String str=new String(“i”)一样吗? 不一定
 System.identityHashCode(str);方法用于获取字符串的引用地址,每个对象都有属于自己的引用地址。
*/


public class Test {
    public static void main(String[] args) {
 //java在jvm内存中编译的时,对象在初始化的开辟内存的时候,如果发现在缓冲区发现有相同实例(内容相同)的对象,则会将引用地址指向该实例对象
        Integer int1=99;
        Integer int2=99;
        System.out.println(int1==int2);//比较的是对象的引用,基本数据类型是比较的值
        System.out.println("int1="+System.identityHashCode(int1));
        System.out.println("int2="+System.identityHashCode(int2));

        String s1="99";
        String s2="99";
        System.out.println(s1==s2); //字符也是比较对象的引用
        System.out.println("s1="+System.identityHashCode(s1));
        System.out.println("s2="+System.identityHashCode(s2));

        //声明两个对象
        String str1="sugar";
        String str2=new String("sugar");//new一个对象,会在堆里开辟一个新的空间

        //比较两个对象的引用地址
        int str1Hashcode=System.identityHashCode(str1);
        int str2Hashcode=System.identityHashCode(str2);

        System.out.println("st1的引用地址="+str1Hashcode);
        System.out.println("st2的引用地址="+str2Hashcode);
        System.out.println(str1==str2);

        //比较两个对象的实体内容
        System.out.println(str1.equals(str2));

        //str1和str2的hashcode值相等,那么他们的hashcode值一定相等
        System.out.println(str1.hashCode()+"=="+str2.hashCode());
    }
}

/**---------------------------------------------------------------------------------------------
结果:
true
int1=951007336
int2=951007336
true
s1=2001049719
s2=2001049719
st1的引用地址=1528902577
st2的引用地址=1927950199
false
true
109792566==109792566
*/

总结

  1. 在类中没有重写hashCode的时候,hashcode值就是对象引用地址
  2. 每一个实例对象都有独立且唯一的内存地址,用来区别每个对象。
  3. hashCode是根据对象属性生成的,用来区别对象内容的不同。
  4. System.identityHashCode()获取引用地址的方法
  5. 以下是关于hashcode的一些规定:
    • 两个对象相等,hashcode一定相等
    • 两个对象不等,hashcode不一定不等
    • hashcode相等,两个对象不一定相等
    • hashcode不等,两个对象一定不等
  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
equals和hashcodeJava两个重要的方法,用于对象的比较和哈希计算。它们之间有以下关系: 1. equals()方法用于判断两个对象是否相等。它是Object类的方法,在需要比较自定义对象时,我们通常需要重写这个方法。equals()方法的默认实现是比较对象的引用是否相等,即比较内存地址。但是,对于自定义的类,我们通常需要根据实际情况重写equals()方法,根据对象的内容进行比较。 2. hashcode()方法用于计算对象的哈希哈希是一个整数,用于标识对象在哈希的位置。在使用哈希表数据结构时,比如HashMap、HashSet等,hashcode()方法用于确定对象在哈希的桶位置。默认情况下,hashcode()方法返回的是对象的内存地址哈希。 3. 根据Java规范,如果两个对象使用equals()方法判断相等,则它们的hashcode()方法必须返回相同的。也就是说,如果两个对象相等,它们的哈希也必须相等。但是,反过来并不成立:如果两个对象的哈希相等,并不一定能说明它们相等。 4. 在重写equals()方法时,通常也需要重写hashcode()方法,以保持equals()和hashcode()之间的一致性。这是因为在哈希表等使用哈希的数据结构,如果两个对象的哈希不相等,它们将被认为是不同的对象,即使equals()方法判断它们相等。 综上所述,equals()和hashcode()是相互关联的方法,用于对象的比较和哈希计算。重写这两个方法需要遵循一定的规范,以确保在使用哈希表等数据结构时能够正常工作。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值