2020年Java面试208题 003-两个对象的hashCode相同,是否equals也一定为true,为什么?

时间: 2021年2月24日星期三  晴

前言

面试是类似演讲演说的过程,网上的资料都是文字的,没法让面试者直观的体验到面试过程,本人已经将相关内容制作成音频节目,有兴趣的朋友可以去喜马拉雅搜索《2020年Java面试208题》收听音频版节目。

《2020年Java面试208题》整理了208个Java程序员的面试题,主要是针对三年以内开发经验的JAVA程序员提出的问题。

题目

这一期要说的是《2020年Java面试208题》的第3题,面试官提了这样一个问题:两个对象的hashCode相同,是否equals也一定为true,为什么?

知识点解析

1 散列表

这个问题里面涉及到一个散列表的概念,我先给解释一下:我们举个例子,图书馆里有100万本书,堆放在一个大仓库里,摆放的倒是整整齐齐的,但是没有做分类管理,有人要找书的话流程是这样的,说出书的名字、作者这些信息,然后在大仓库里面一本本的找,可以想象这个任务是多么的艰巨而乏味。怎么解决这个问题呢?

办法1:管理员把图书按照类型、年代、书名首字母做了一个分类,每次找书可以直接到那个分类下,根据书名首字母进行查找,范围一下缩小到不到1万本书里面去查找了。

 

办法2:管理员按照书名拼音全拼的顺序来摆放图书,找图书的时候,比如《钢铁是怎么炼成的》这本书,拼音是gangtieshizenyanglianchengde,只要找到g区、再找a字母、再找n字母,很快就找到了。

 

办法3:后来管理员又想了一个办法,把书名字符串计算为一个整数,然后每个整数做一个小盒子,然后把图书放入对应的小盒子里面,这样每个小盒子里面都只放了1本书,最多两三本。之后管理员把小盒子按从小到大的顺序摆放。这样每次找书的时候只要计算一下书名对应的整数,然后在仓库里找到对应位置的盒子,就很快找到了。

 

默认情况下的方式,就是傻傻的一本本的比对,恐怕现实中谁都不会这么做,但在计算机编程的时候,很多初级程序员都是这么写的,就是循环查找比对。

第1种办法基本现在图书馆都是基于这个思路来管理的,这种按照类型、年代、书名首字母这样方式分类的在现实世界里容易操作。

第2种方法其实在现实世界里没有人会这么操作,但作为计算机中的管理就是一个不错的方法了,但编码过程存在复杂性,毕竟计算拼音的过程就比较麻烦。

第3种方法中,将字符串的书名全部计算为一个整数,然后做成一个个的小盒子顺序排放,这种方式在计算机中被大量的使用,这就叫散列表。为啥叫散列呢?因为这样的书名计算结果一定不是一个挨着一个的顺序数字,而是中间会空出很多数字,在数学形式上是非连续的,术语叫离散的,因此表示成离散的、形成顺序的列,就叫散列表。

 

比如Java中的HashMap,总是说HashMap查询速度快,根本原因就是因为HashMap中的内容是按照第3种方法排放的。在HashMap中排放书的话之前已经解释过了,用书名计算出一个整数就行了,但是HashMap的key是支持各种各样的类的,这个整数的计算咋处理呢?Java就提供了一个hashCode方法,程序员根据自己设计的类的需要,想咋实现咋实现,只要最后计算出一个相对离散的整数就可以了。

 

这下相信你已经理解hashCode方法的设计目的了。

 

equals方法也是完全自定义的方法,由程序员自行实现,当然Java提供的类是由Sun公司或Oracle公司的程序员实现的equals方法。这个方法的设计目的是只要两个对象中的值有一点儿不相同,返回就是false,只有两个对象中的值完全一样才会返回true。

 

对于hashCode和equals方法的功能解释完了,接下来就由我来演示一下面试:

演示面试

你好,面试官。

 

1 我先说答案,hashCode相同,equals方法不一定为true。

 

2 在解释为什么之前,我先说一下hashCode方法和equals方法的定义。

我先说说hashCode方法,这个方法是Object类提供的,所有的类都重写了这个方法,这个方法的目的是计算出对象的散列码。这个不同类型对象的hashCode方法的实现是不一样的,比如整型对象,就是直接拿到整数对象的值作为散列码,那么就可以知道了,值不相同的整型对象hashCode也不相同,值相同的整型对象hashCode也一样。

浮点数是将浮点数值计算为唯一的整型作为散列码,因此浮点数的hashCode也不会重复。

Double类型的对象的hashCode的计算方法是将double值计算为长整型,然后只取高位的32位的整型,因为Double类型在内存中占64位,可以直接转换为唯一的Long长整型,因为长整型也是64位的。但hashCode只能返回32位的整型,因此Double类型计算出64位的长整型后,将这个长整型再舍弃一部分只保留高位的32位的整型作为散列码。

 

通过这样的描述可以看出,整型Integer的hashCode一定不会重复,浮点数Float的hashCode也不会重复,但Double类型的hashCode很多时候都会重复,因为总共有可能有2的64次方个数字,最后只允许生成2的32次方个散列码,那其中一定有大量的重复。

 

其实不用这么复杂的分析,简单的想象一下就能得到值不相同的对象的hashCode值是有可能相同的,因为hashCode的返回值是整型的,也就是说hashCode的返回值的可能性只有2的32次方个整数那么多,而现实中各种类型的数据的值绝对不只有这么少。

 

对于equals的定义是值不相同的对象,equals方法一定为False。

 

所以,可以得出结论两个对象的hashCode相同,equals不一定为true。

 

3 hashCode方法设计出来本来就不是追求唯一性的,而是作为散列表的散列码生成算法的,只要分散够均匀就行了。

 

好了,以上就是我的演示面试,不知道是否让你满意。

 

自我点评:

本题目依旧是明知跑题也要跑题,但跑的又不是很远,目的就是尽量多讲一些时间,你讲的多多的,让面试官讲的少少的,你就是赢家了。美国参议院有一种制度,当议员对某项议题存在不同意见的时候,允许议员在国会议事厅发表演讲阐述自己的观点,只要议员能讲,就可以一直讲下去,但不允许吃东西、喝水、睡觉,也不允许上厕所,也不允许坐着,当然也不能停下来不说,有的州甚至不限制讲演内容,议员甚至可以在台上念整本的圣经,这种制度的英文是filibuster,音译成中文叫费力把事拖。程序员面试的时候也可以采用这样的战术。

 

另外,在讲解的时候先切入主题,把自己的观点先表达清楚,然后再层层剖析,这样回答问题条例会更清晰。

 

下期再见

本人技术有限,凭借热情希望给程序员面试提供一点帮助,必有不足之处,希望业内人士积极批评指正,在留言区留言就是对我最大的鼓励。希望批评中肯一些,不要存在人身攻击,更不要波及到家人,善意中肯的批评我都能够虚心接受。 我们下期再见。

面试包括以下十九部分:Java 基础、容器、多线程、反射、对象拷贝、Java Web 模块、异常、网络、设计模式、Spring/Spring MVC、Spring Boot/Spring Cloud、Hibernate、Mybatis、RabbitMQ、Kafka、Zookeeper、MySql、Redis、JVM 。 目录: 一、Java 基础 1.JDK 和 JRE 有什么区别? 2.== 和 equals 的区别是什么? 3.两个对象hashCode()相同,则 equals()也一定true,对吗? 4.final 在 java 有什么作用? 5.java 的 Math.round(-1.5) 等于多少? 6.String 属于基础的数据类型吗? 7.java 操作字符串都有哪些类?它们之间有什么区别? 8.String str="i"与 String str=new String(“i”)一样吗? 9.如何将字符串反转? 10.String 类的常用方法都有那些? 11.抽象类必须要有抽象方法吗? 12.普通类和抽象类有哪些区别? 13.抽象类能使用 final 修饰吗? 14.接口和抽象类有什么区别? 15.java IO 流分为几种? 16.BIO、NIO、AIO 有什么区别? 17.Files的常用方法都有哪些? 二、容器 18.java 容器都有哪些? 19.Collection 和 Collections 有什么区别? 20.List、Set、Map 之间的区别是什么? 21.HashMap 和 Hashtable 有什么区别? 22.如何决定使用 HashMap 还是 TreeMap? 23.说一下 HashMap 的实现原理? 24.说一下 HashSet 的实现原理? 25.ArrayList 和 LinkedList 的区别是什么? 26.如何实现数组和 List 之间的转换? 27.ArrayList 和 Vector 的区别是什么? 28.Array 和 ArrayList 有何区别? 29.在 Queue poll()和 remove()有什么区别? 30.哪些集合类是线程安全的? 31.迭代器 Iterator 是什么? 32.Iterator 怎么使用?有什么特点? 33.Iterator 和 ListIterator 有什么区别? 34.怎么确保一个集合不能被修改? 三、多线程 35.并行和并发有什么区别? 36.线程和进程的区别? 37.守护线程是什么? 38.创建线程有哪几种方式? 39.说一下 runnable 和 callable 有什么区别? 40.线程有哪些状态? 41.sleep() 和 wait() 有什么区别? 42.notify()和 notifyAll()有什么区别? 43.线程的 run()和 start()有什么区别? 44.创建线程池有哪几种方式? 45.线程池都有哪些状态? 46.线程池 submit()和 execute()方法有什么区别? 47.在 java 程序怎么保证多线程的运行安全? 48.多线程锁的升级原理是什么? 49.什么是死锁? 50.怎么防止死锁? 51.ThreadLocal 是什么?有哪些使用场景? 52.说一下 synchronized 底层实现原理? 53.synchronized 和 volatile 的区别是什么? 54.synchronized 和 Lock 有什么区别? 55.synchronized 和 ReentrantLock 区别是什么? 56.说一下 atomic 的原理? 四、反射 57.什么是反射? 58.什么是 java 序列化?什么情况下需要序列化? 59.动态代理是什么?有哪些应用? 60.怎么实现动态代理? 五、对象拷贝 61.为什么要使用克隆? 62.如何实现对象克隆? 63.深拷贝和浅拷贝区别是什么? 六、Java Web 64.jsp 和 servlet 有什么区别? 65.jsp 有哪些内置对象?作用分别是什么? 66.说一下 jsp 的 4 种作用域? 67.session 和 cookie 有什么区别? 68.说一下 session 的工作原理? 69.如果客户端禁止 cookie 能实现 session 还能用吗? 70.spring mvc 和 struts 的区别是什么? 71.如何避免 sql 注入? 72.什么是 XSS 攻击,如何避免? 73.什么是 CSRF 攻击,如何避免? 七、异常 74.throw 和 throws 的区别? 75.final、finally、finalize 有什么区别? 76.try-catch-finally 哪个部分可以省略? 77.try-catch-finally ,如果 catch return 了,finally 还会执行吗? 78.常见的异常类有哪些? 八、网络 79.http 响应码 301 和 302 代表的是什么?有什么区别? 80.forward 和 redirect 的区别? 81.简述 tcp 和 udp的区别? 82.tcp 为什么要三次握手,两次不行吗?为什么? 83.说一下 tcp 粘包是怎么产生的? 84.OSI 的七层模型都有哪些? 85.get 和 post 请求有哪些区别? 86.如何实现跨域? 87.说一下 JSONP 实现原理? 九、设计模式 88.说一下你熟悉的设计模式? 89.简单工厂和抽象工厂有什么区别? 十、Spring/Spring MVC 90.为什么要使用 spring? 91.解释一下什么是 aop? 92.解释一下什么是 ioc? 93.spring 有哪些主要模块? 94.spring 常用的注入方式有哪些? 95.spring 的 bean 是线程安全的吗? 96.spring 支持几种 bean 的作用域? 97.spring 自动装配 bean 有哪些方式? 98.spring 事务实现方式有哪些? 99.说一下 spring 的事务隔离? 100.说一下 spring mvc 运行流程? 101.spring mvc 有哪些组件? 102.@RequestMapping 的作用是什么? 103.@Autowired 的作用是什么? 十一、Spring Boot/Spring Cloud 104.什么是 spring boot? 105.为什么要用 spring boot? 106.spring boot 核心配置文件是什么? 107.spring boot 配置文件有哪几种类型?它们有什么区别? 108.spring boot 有哪些方式可以实现热部署? 109.jpa 和 hibernate 有什么区别? 110.什么是 spring cloud? 111.spring cloud 断路器的作用是什么? 112.spring cloud 的核心组件有哪些? 十二、Hibernate 113.为什么要使用 hibernate? 114.什么是 ORM 框架? 115.hibernate 如何在控制台查看打印的 sql 语句? 116.hibernate 有几种查询方式? 117.hibernate 实体类可以被定义为 final 吗? 118.在 hibernate 使用 Integer 和 int 做映射有什么区别? 119.hibernate 是如何工作的? 120.get()和 load()的区别? 121.说一下 hibernate 的缓存机制? 122.hibernate 对象有哪些状态? 123.在 hibernate getCurrentSession 和 openSession 的区别是什么? 124.hibernate 实体类必须要有无参构造函数吗?为什么? 十三、Mybatis 125.mybatis #{}和 ${}的区别是什么? 126.mybatis 有几种分页方式? 127.RowBounds 是一次性查询全部结果吗?为什么? 128.mybatis 逻辑分页和物理分页的区别是什么? 129.mybatis 是否支持延迟加载?延迟加载的原理是什么? 130.说一下 mybatis 的一级缓存和二级缓存? 131.mybatis 和 hibernate 的区别有哪些? 132.mybatis 有哪些执行器(Executor)? 133.mybatis 分页插件的实现原理是什么? 134.mybatis 如何编写一个自定义插件? 十四、RabbitMQ 135.rabbitmq 的使用场景有哪些? 136.rabbitmq 有哪些重要的角色? 137.rabbitmq 有哪些重要的组件? 138.rabbitmq vhost 的作用是什么? 139.rabbitmq 的消息是怎么发送的? 140.rabbitmq 怎么保证消息的稳定性? 141.rabbitmq 怎么避免消息丢失? 142.要保证消息持久化成功的条件有哪些? 143.rabbitmq 持久化有什么缺点? 144.rabbitmq 有几种广播类型? 145.rabbitmq 怎么实现延迟消息队列? 146.rabbitmq 集群有什么用? 147.rabbitmq 节点的类型有哪些? 148.rabbitmq 集群搭建需要注意哪些问? 149.rabbitmq 每个节点是其他节点的完整拷贝吗?为什么? 150.rabbitmq 集群唯一一个磁盘节点崩溃了会发生什么情况? 151.rabbitmq 对集群节点停止顺序有要求吗? 十五、Kafka 152.kafka 可以脱离 zookeeper 单独使用吗?为什么? 153.kafka 有几种数据保留的策略? 154.kafka 同时设置了 7 天和 10G 清除数据,到第五天的时候消息达到了 10G,这个时候 kafka 将如何处理? 155.什么情况会导致 kafka 运行变慢? 156.使用 kafka 集群需要注意什么? 十六、Zookeeper 157.zookeeper 是什么? 158.zookeeper 都有哪些功能? 159.zookeeper 有几种部署模式? 160.zookeeper 怎么保证主从节点的状态同步? 161.集群为什么要有主节点? 162.集群有 3 台服务器,其一个节点宕机,这个时候 zookeeper 还可以使用吗? 163.说一下 zookeeper 的通知机制? 十七、MySql 164.数据库的三范式是什么? 165.一张自增表里面总共有 7 条数据,删除了最后 2 条数据,重启 mysql 数据库,又插入了一条数据,此时 id 是几? 166.如何获取当前数据库版本? 167.说一下 ACID 是什么? 168.char 和 varchar 的区别是什么? 169.float 和 double 的区别是什么? 170.mysql 的内连接、左连接、右连接有什么区别? 171.mysql 索引是怎么实现的? 172.怎么验证 mysql 的索引是否满足需求? 173.说一下数据库的事务隔离? 174.说一下 mysql 常用的引擎? 175.说一下 mysql 的行锁和表锁? 176.说一下乐观锁和悲观锁? 177.mysql 问排查都有哪些手段? 178.如何做 mysql 的性能优化? 十八、Redis 179.redis 是什么?都有哪些使用场景? 180.redis 有哪些功能? 181.redis 和 memecache 有什么区别? 182.redis 为什么是单线程的? 183.什么是缓存穿透?怎么解决? 184.redis 支持的数据类型有哪些? 185.redis 支持的 java 客户端都有哪些? 186.jedis 和 redisson 有哪些区别? 。。。。。。。。。。。。。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

小崔爱读书

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值