java string hash 比较_Java String 的equals, == , hascode的区别

1、equals 和 ==

==在java中是比较引用的,即在内存中的地址。而String的equals()是比较字符串的内容

先看这几句代码

String s1 = "String";

String s2 = "String";

System.out.println(s1 == s2 );

System.out.println(s1.equals(s2));

结果为:

true

true

这个结果说明什么呢?

==在java中是比较引用的,即在内存中的地址。而String的equals()是比较字符串的内容。

第二个true无可争议,因为s1和s2的内容显然 相同。

第一个true呢?

这说明s1和s2的地址相同,s1和s2的地址为什么会相同?

查阅资料,发现 :

String s1 = "String";

这种方式,java首先会在缓冲区查找是否有"String"这个常量对象,有就直接将其地址赋给s1,没有就创建一个"String",然后将其赋给s1;然后

String s2 = "String";

java同样会在缓冲区中查找"String",这次能查找到了,因为s1创建了一个"String",所以会将其地址赋给s2,如此,s1和s2便有了相同的地址。

~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~

OK。

再看这一段代码

String s3 = new String("String");

String s4 = new String("String");

System.out.println(s3 == s4);

System.out.println(s3.equals(s4));

结果为:

false

true

第二个true同样无争议。

第一个false说明s3和s4不是指向同一地址。

查阅资料发现,

String s3 = new String("String");会直接在内存中开辟一个空间存储一个"String",并讲引用赋给s3;

同样 String s4 = new String("String");也会开辟一个空间,降低至给s4;

所以s3和s4的地址不一样。

System.out.println(s3 == s4); 会打印出false。

///*/

关于String的equals方法:

这是直接从jdk源码复制的equals方法代码:

public boolean equals(Object anObject) {

if (this == anObject) {

return true;

}

if (anObject instanceof String) {

String anotherString = (String)anObject;

int n = count;

if (n == anotherString.count) {

char v1[] = value;

char v2[] = anotherString.value;

int i = offset;

int j = anotherString.offset;

while (n-- != 0) {

if (v1[i++] != v2[j++])

return false;

}

return true;

}

}

return false;

}

java的String是用字符数组模拟的。

从上面的代码可以看出String的equals()的工作原理——

1、首先比较引用,如果引用相同,返回true;

2、比较类型,如果不是比较的不是String对象,返回false;

3、比较长度,字符串长度不等时,返回false;

4、逐个字符比较两个字符串,遇到不一样的字符,返回false;

5、到最后都一致,返回ture;

2、hashcode

在这里主要是讲==与hashCode()之间的区别。

String str1 = "nihao" ;

String str2 = "nihao" ;

String str3 = new String("nihao");

String str4 = new String("nihao");

System.out.println("str1==str2: " + (str1==str2));    // true

System.out.println("str1==str3: " + (str1==str3));    // false

System.out.println("str3==str4: " + (str3==str4));   // false

System.out.println("HashCode:" + str3.hashCode());  // HashCode:104818427

System.out.println("HashCode:" + str4.hashCode());  // HashCode:104818427

System.out.println(str3.hashCode() == str4.hashCode());  // true

System.out.println(str3.equals(str4));  // true

第一个true,是因为str1和str2是同一个变量,这个变量在变量池中。

第一个false,是因为str1在变量池中,而str3在堆栈中,所以是不同。

第二个false,通过new创建的String,是两个不同的对象。

虽然str3和str4是两个不同的对象,但是其hashCode值是一样的,这就和hashCode()函数有关了,下面是hashCode()的源函数:

public int hashCode() {

int h = hash;

if (h == 0) {

int off = offset;

char val[] = value;

int len = count;

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

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

}

hash = h;

}

return h;

}

}

在String类中,value是该字符串被转换成数组后的对象。由于offset总是为0,所以h一开始就是0。于是只要value是一样的,所 以hashcode一定是一样的。于是可以肯定的说,==在判断对象时,其实是根据对象在堆栈中的地址判断对象是不是一样,而不是根据hashcode 值。

Java String中的HashCode和equal

1. hashSet中比较是否重复的依据是a.hasCode()=b.hasCode() && a.equals(b)

2. String的hashCode依据: 以依赖于char[i]的int值以和char[i]的排列序的算法计算出的。不依赖String的ref.

3. String的equals依据: a==b || ( a.length=b.length && { a[i]=b[i] } )

4. 只有用a==b时比校的才是比校的ref,也就是说这时才是比校是a与b是不是同一个对象

5. 结论: 两个不同ref的String可能会被认为是集合中的同一个元素。

关于String.hashCode

String.hashCode(),提供将任意长的字符串哈希成一个整数。

在Java中,整型数是32位的,也就是说最多有2^32= 4294967296个整数,将任意一个字符串,经过hashCode计算之后,得到的整数应该在这4294967296数之中。那么,最多有 4294967297个不同的字符串作hashCode之后,肯定有两个结果是一样的(这个可以用鸽巢定理来解释,hehe)。

下面的两组,其hashCode就相等了:

http://pic.bandaonews.com/PicView.aspx?id=37219

http://tech.163.com/05/0829/09/1SAIIRG8000915BD.html

http://news.sina.com.cn/c/2005-06-08/10076113241s.shtml

http://news.edw.com.cn/show.aspx?id=3490&cid=25

当用到这样的方法来得到字符串的哈希值,并且此哈希值可能作为系统中的一个比较重要的参数的时候,应该充分考虑到哈希值可能重复的后果并做好一定的容错处理

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值