哈希码总结:

哈希码

一 哈希码

在Java中,哈希码代表了对象的一种特征,例如我们判断某两个字符串是否==,如果其哈希码相等,则这两个字符串是相等的。其次,哈希码是一种数据结构的算法。常见的哈希码的算法有:

1:Object类的hashCode.返回对象的内存地址经过处理后的结构,由于每个对象的内存地址都不一样,所以哈希码也不一样。

2:String类的hashCode.根据String类包含的字符串的内容,根据一种特殊算法返回哈希码,只要字符串内容相同,返回的哈希码也相同。
3:Integer类,返回的哈希码就是Integer对象里所包含的那个整数的数值,例如Integer i1=new Integer(100),i1.hashCode的值就是100 。由此可见,2个一样大小的Integer对象,返回的哈希码也一样。

二 Object对象默认的toString()

假如.直接输出一个实例对象,出现一串字符串,代表什么?

直接输出一个类的对象的时候,会调用这个类的toString()方法,这个方法有些类是覆盖了的,比如String,Integer。你自己写的类没有覆盖这个方法的话就是继承Object类的这个方法,Object中toString()方法的实输出格式是这样的getClass().getName() + “@” + Integer.toHexString(hashCode()) 后面跟的是这个类的哈希码,如果你希望这个类打印出来输出你希望的格式,你就要覆盖这个、toString方法。
测试:

package new_start1;
public class Test1 {
class Person
{
public String name;
public Person(String n)
{
this.name=n;
}
public Person(){}
}
public static void change(Person a)//改变对象a的name值
{
a.name=“haha”;
}
public static void main(String[] args) {
Test1 t=new Test1();
Person p=t.new Person(“zhangsan”); //实例一个对象p
Person a=t.new Person();//又实例一个对象a
System.out.println(“未赋值前,两者的哈希码是不相同的:”);
System.out.println(“a.hashCode=”+a.hashCode()+" "+“p.hashCode=”+p.hashCode());
System.out.println(“a.toString()=”+a.toString());
System.out.println(“p.toString()=”+p.toString());
/*
未赋值前,两者的哈希码是不相同的:
a.hashCode=366712642 p.hashCode=1829164700
a.toString()=new_start1.Test1 P e r s o n @ 15 d b 9742 p . t o S t r i n g ( ) = n e w s t a r t 1. T e s t 1 Person@15db9742 p.toString()=new_start1.Test1 Person@15db9742p.toString()=newstart1.Test1Person@6d06d69c
/
a=p;
System.out.println(“赋值后,两者的哈希码相同:”);
System.out.println(“a.hashCode=”+a.hashCode()+" "+“p.hashCode=”+p.hashCode());
System.out.println(“a.toString()=”+a.toString());
System.out.println(“p.toString()=”+p.toString());
/

赋值后,两者的哈希码相同:
a.hashCode=1829164700 p.hashCode=1829164700
a.toString()=new_start1.Test1 P e r s o n @ 6 d 06 d 69 c p . t o S t r i n g ( ) = n e w s t a r t 1. T e s t 1 Person@6d06d69c p.toString()=new_start1.Test1 Person@6d06d69cp.toString()=newstart1.Test1Person@6d06d69c
*/
}
}

点赞 1
哈希码是一种数据结构的算法。

哈希码具体是什么?

答:hashCode是jdk根据对象的地址或者字符串或者数字算出来的int类型的数值

常见的哈希码的算法有:

1:Object类的hashCode.返回对象的内存地址经过处理后的结构,由于每个对象的内存地址都不一样,所以哈希码也不一样。

2:String类的hashCode.根据String类包含的字符串的内容,根据一种特殊算法返回哈希码,只要字符串内容相同,返回的哈希码也相同。

3:Integer类,返回的哈希码就是Integer对象里所包含的那个整数的数值,例如Integer i1=new Integer(100),i1.hashCode的值就是100 。

跟踪Object类的native方法hashCode方法从jvm源码中得到了下面的一些内容,供参考。

Object中hashCode方法是一个本地方法:public native inthashCode();

对于Java HotSpot VM,首先介绍一个概念就是对象的header,

每个对象都会有一个header,header由两个机器字表示(8个字节对于32位架构,16个字节对于64位架构)。

header的第一个字中有7位用做同步及垃圾收集,另外25位存储对象的hash码。

header的第二个字存储指向对应Class对象的指针(Class对象用来保存类的元数据信息及方法表)。

hashcode 作用 :对象实例的唯一标识

在同一运行环境下 hashcode 的值是唯一的

就是两个不同实例其hashcode在同一运行环境绝对不一样 主要用来区分 两个实例在物理上是不是同一个对象。

如:

string a =“111”;

string b =“111”;

a和b 的hashcode是一样的。

其原因是java的字符串池优化原因,你声明一个字符串时JVM会先去查找 字符串池 是否有相同字符串有将已经有的字符串对象的引用返回 而不是新生成一个字符对象到内存没有 新生成 并将其引用放入字符串池 如此循环所以他们是同一个对象 其hashcode也一样

至于这个

String a = newString(“i love you”);

String b = newString(“i love you”);

如果hashcode也一样的话 那他们应该也是类似的优化。jdk5 好像不一样吧?

hashcode()是要在容器里面的MAP这个才能体现其价值,在MAP里面要是重写equals,就要重写hashcode的方法,只要equals为真,那么hashcode也应该一样。平时其他的使用hashcode没什么大用。

对于Object对象来说,不同的Object对象的hashcode是不同的,它们返回的是对象的地址,equals返回的也是对象的地址。

所以在自己定义的类中如果要添加到集合对象中,最好是要重写hashcode和equals方法,不然会自动继承自Object类中的两个方法根据对象地址来判断。在重写自己定义的类时,通常是在类中的根据某个值如name.hashcode();来进行判断。

一般来讲,equals这个方法是给用户调用的,如果你想判断2个对象是否相等,你可以重写equals方法,然后在代码中调用,就可以判断他们是否相等了。简单来讲,equals方法主要是用来判断从表面上看或者从内容上看,2个对象是不是相等。举个例子,有个学生类,属性只有姓名和性别,那么我们可以认为只要姓名和性别相等,那么就说这2个对象是相等的。

hashcode方法一般用户不会去调用,比如在hashmap中,由于key是不可以重复的,他在判断key是不是重复的时候就判断了hashcode这个方法,而且也用到了equals方法。这里不可以重复是说equals和hashcode只要有一个不等就可以了!所以简单来讲,hashcode相当于是一个对象的编码,就好像文件中的md5,他和equals不同就在于他返回的是int型的,比较起来不直观。我们一般在覆盖equals的同时也要覆盖hashcode,让他们的逻辑一致。举个例子,还是刚刚的例子,如果姓名和性别相等就算2个对象相等的话,那么hashcode的方法也要返回姓名的hashcode值加上性别的hashcode值,这样从逻辑上,他们就一致了。

要从物理上判断2个对象是否相等,用==就可以了。

"=="和equals方法究竟有什么区别?

操作符专门用来比较两个变量的值是否相等,也就是用于比较变量所对应的内存中所存储的数值是否相同,要比较两个基本类型的数据或两个引用变量是否相等,只能用操作符。

如果一个变量指向的数据是对象类型的,那么,这时候涉及了两块内存,对象本身占用一块内存(堆内存),变量也占用一块内存,例如Objet obj = new Object();变量obj是一个内存,new Object()是另一个内存,此时,变量obj所对应的内存中存储的数值就是对象占用的那块内存的首地址。对于指向对象类型的变量,如果要比较两个变量是否指向同一个对象,即要看这两个变量所对应的内存中的数值是否相等,这时候就需要用==操作符进行比较。

equals方法是用于比较两个独立对象的内容是否相同,就好比去比较两个人的长相是否相同,它比较的两个对象是独立的。例如,对于下面的代码:

String a=new String(“foo”);

String b=new String(“foo”);

两条new语句创建了两个对象,然后用a,b这两个变量分别指向了其中一个对象,这是两个不同的对象,它们的首地址是不同的,即a和b中存储的数值是不相同的,所以,表达式a==b将返回false,而这两个对象中的内容是相同的,所以,表达式a.equals(b)将返回true。

在实际开发中,我们经常要比较传递进行来的字符串内容是否等,例如,String input = …;input.equals(“quit”),许多人稍不注意就使用==进行比较了,这是错误的,随便从网上找几个项目实战的教学视频看看,里面就有大量这样的错误。记住,字符串的比较基本上都是使用equals方法。

如果一个类没有自己定义equals方法,那么它将继承Object类的equals方法,Object类的equals方法的实现代码如下:

boolean equals(Object o){

return this==o;

}

这说明,如果一个类没有自己定义equals方法,它默认的equals方法(从Object类继承的)就是使用操作符,也是在比较两个变量指向的对象是否是同一对象,这时候使用equals和使用会得到同样的结果,如果比较的是两个独立的对象则总返回false。如果你编写的类希望能够比较该类创建的两个实例对象的内容是否相同,那么你必须覆盖equals方法,由你自己写代码来决定在什么情况即可认为两个对象的内容是相同的。

  • 2
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 使用哈希表来构建电话号系统的思路是:首先,需要映射电话号与对应的客户信息。可以使用哈希表将电话号映射到客户信息,这样在查找客户信息时,可以通过电话号快速查找到客户信息。其次,哈希表可以快速定位客户的电话号,以便客服人员能够快速查找到客户的电话号。最后,哈希表可以帮助查找电话号是否已经存在,以免发生重复的电话号的情况。 ### 回答2: 构建电话号系统,可以使用哈希表来实现。哈希表是一种快速查找的数据结构,适合用于存储和检索大量的电话号。 首先,我们需要确定电话号的存储结构。可以将每个电话号作为键,将相应的联系人信息作为值存储在哈希表中。 接下来,需要确定哈希函数的选择。哈希函数的作用是将电话号转换为哈希表中的索引。一个好的哈希函数应该尽可能均匀地将电话号分布在哈希表中,并且能够避免冲突。 一种常见的哈希函数是取电话号的后几位数字作为索引。例如,可以将电话号的后四位作为索引,这样相似的电话号可能会有相同的索引,但是哈希表能够处理冲突。 在插入电话号时,我们将电话号作为键,联系人信息作为值,使用哈希函数将电话号转换为哈希表中的索引,并将对应的键值对插入到哈希表中。 在查找电话号时,我们将要查找的电话号作为输入,使用相同的哈希函数将其转换为哈希表中的索引,然后在该索引处查找对应的键值对。如果找到了对应的键值对,即可得到联系人信息。如果没有找到,则说明该电话号不存在于电话号系统中。 此外,还可以考虑处理哈希冲突的方法。一种常见的方法是使用链表,即在哈希表中的每个索引处维护一个链表,当有多个键值对映射到同一个索引时,将新的键值对插入到链表的末尾。 总结来说,构建电话号系统的思路是确定存储结构,选择合适的哈希函数,使用哈希表存储电话号和联系人信息,并处理哈希冲突。这样就可以实现快速的电话号存储和检索功能。 ### 回答3: 构建电话号系统可以使用哈希表来实现快速的号查找和存储。下面是一个可以用哈希表构建电话号系统的思路: 1. 定义哈希表:创建一个空的哈希表,作为电话号系统的存储容器。可以使用数组和链表的方式实现哈希表。 2. 哈希函数设计:设计一个合适的哈希函数,用于将电话号转换为哈希表中的索引。可以根据电话号的特征进行设计,比如将电话号的每个数字相加取模,或者使用字符串的哈希函数。 3. 添加电话号:将电话号作为键,将对应的姓名或其他相关信息作为值,使用哈希函数计算出索引位置,将电话号和相关信息添加到哈希表中对应的位置。 4. 查找电话号:当需要查找电话号时,使用哈希函数计算出对应的索引位置,然后在该位置的链表中查找相关信息。 5. 更新电话号:如果电话号的相关信息需要更新,可以先根据电话号找到对应的索引位置,然后更新链表中的值。 6. 删除电话号:如果要删除某个电话号的相关信息,可以先根据电话号找到对应的索引位置,然后在链表中删除该节点。 哈希表通过将电话号转换为索引,可以显著减少查找和插入的时间复杂度,从而提高了电话号系统的效率。但需要注意的是,哈希函数的设计要避免冲突,避免不同电话号映射到同一个索引位置,可以使用开放地址法或链地址法解决哈希冲突的问题。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值