【Java】哈希值相同的变量_对象的地址值也相同吗?

前言

  “Java中直接打印对象,输出的是对象的地址值(对象的类没有重写toString方法)”,这是初学者在学Java基础语言的时候,大多数教程都是这么介绍的,初学者也会认为这就是对象在内存中存储的位置,如输出的地址值为:

1
cn.imcyc.person.Person@11ff03

但是,其实这种说法是有混淆性的,实际上,打印对象输出的“地址值”并不是对象的物理存储地址,@之后的“11ff03”只是这个对象的哈希值的十六进制。

  实际上,Java语言中其实是看不到变量/对象真正的物理地址值的,只能通过hashCode方法查到变量/对象的哈希值,在极少数的情况下,变量/对象的哈希值会重复,也就是“哈希碰撞”、“哈希混淆”现象。如果初学者看到两个不同数据的类型/对象打印出的“包名.类名@十六进制哈希值”是一样的,就会误以为这两个类型/对象的地址值是相同的,这种想法是错误的。

  下面介绍一种“哈希冲突”的情况。

哈希值相同的String类对象

1
2
3
4
String a = "重地";
String b = "通话";
System.out.println(a.hashCode());
System.out.println(b.hashCode());

上面代码的执行结果是:

1
2
1179395
1179395

字符串类对象a和b的哈希值是相同的,但事实上两者的物理存储地址是不同的。那么如果将“重地”和“通话”作为字符串存储到对象中,对象的哈希值会一样吗?

哈希值相同的自定义类对象

  新建一个Person类,有一个私有字符串数据成员,两个构造函数,如下:

1
2
3
4
5
6
7
8
9
10
11
public class Person {

private String name;

public Person() {
}

public Person(String name) {
this.name = name;
}
}

  主程序执行如下代码:

1
2
3
4
5
6
Person c = new Person("重地");
Person d = new Person("通话");
System.out.println(c);
System.out.println(c.hashCode());
System.out.println(d);
System.out.println(d.hashCode());

执行结果:

1
2
3
4
cn.imcyc.person.Person@75412c2f
1967205423
cn.imcyc.person.Person@282ba1e
42121758

  可以看到,对象c和d的哈希值不同的。

拓展:重写Person类的hashCode方法

  重写了hashCode方法的Person类代码为:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
public class Person {
private String name;

@Override
public int hashCode() {
return name != null ? name.hashCode() : 0;
}

public Person() {
}

public Person(String name) {
this.name = name;
}
}

上面重写的hashCode的方法是IDEA默认生成的,可以看到就是把数据成员name的哈希值拿来当对象的哈希值,再次执行主程序的代码,输出为:

1
2
3
4
cn.imcyc.person.Person@11ff03
1179395
cn.imcyc.person.Person@11ff03
1179395

可以看到这时的c和d两个对象的哈希值就相同了,但是能说对象c和对象d的地址值相同吗?显然是不能的,对象c和对象d的物理存储地址一定是不同的。

相关文章推荐

  【Java】超严谨论证:为什么自定义类对象存储到Set集合中,要重写hashCode方法和equals方法

————————— 本文结束 感谢您的阅读 —————————
谢谢你请我喝咖啡ლↂ‿‿ↂლ(支付宝扫一扫即可领红包, 消费时可抵现! 你省钱, 我赚钱, 多谢支持~)