互联网大厂Java面试:谢飞机的搞笑面试故事
第一轮:Java基础以及集合框架
面试官:“谢飞机,来吧,第一个问题,Java中的HashMap是线程安全的吗?如果要实现线程安全,你会怎么做?”
谢飞机(信心满满):“当然不是线程安全的啊!线程安全的话,用Hashtable就行了!”
面试官:“不错,回答得很快。那你再谈谈HashMap的底层实现吧?”
谢飞机(挠头):“呃,就是……呃,数组加链表吧?”
面试官:“嗯,好,那我们下一个问题:ArrayList和LinkedList的区别是什么?”
谢飞机(瞪大眼睛):“一个是数组实现,一个是链表实现!”
面试官:“详细点?”
谢飞机(硬着头皮):“ArrayList是动态数组,可以扩容;LinkedList是双向链表,可以随便插入删除。”
面试官:“好吧,暂时过关。”
第二轮:多线程与线程池
面试官:“我们公司用到了线程池,那你能说说线程池的工作原理吗?”
谢飞机:“线程池啊,就是……创建几个线程,然后……用完了再放回去?”
面试官:“说得还行,那线程池的核心参数有哪些?”
谢飞机(开始胡说):“呃,核心参数啊,比如线程池的大小、线程的名字、线程的……呃,线程的颜色?”
面试官(冷漠脸):“线程的颜色?那好吧,我们换个问题。你了解JUC中的ReentrantLock吗?”
谢飞机:“ReentrantLock?就是一个可以重入的锁吧?”
面试官:“再详细点?”
谢飞机:“呃,可以tryLock……然后……然后就没了!”
面试官:“行,那我们继续下一轮。”
第三轮:分布式与中间件
面试官:“我们公司用Dubbo做服务治理,那你能简单讲讲Dubbo的核心架构吗?”
谢飞机:“Dubbo啊,就是……就是一个RPC框架吧?”
面试官:“再具体点?”
谢飞机:“呃,有注册中心、服务提供者和服务消费者……然后……然后就没了!”
面试官:“好,那我们再问最后一个,RabbitMQ的消息确认机制是怎么实现的?”
谢飞机:“RabbitMQ啊,消息确认……就是发消息,然后收消息?”
面试官(扶额):“嗯……谢飞机,你回去等通知吧。”
面试问题答案详解
HashMap是线程安全的吗?如何实现线程安全?
- HashMap本身不是线程安全的。如果需要实现线程安全,可以使用
Collections.synchronizedMap方法对其进行包装,或者使用ConcurrentHashMap。
- HashMap本身不是线程安全的。如果需要实现线程安全,可以使用
HashMap的底层实现
- HashMap的底层是由数组和链表组成的。在Java 8之后,当链表长度超过阈值(默认8)时,会将链表转换为红黑树。
ArrayList和LinkedList的区别
- ArrayList是基于动态数组实现的,支持随机访问,查询速度快,但插入删除效率低。
- LinkedList是基于双向链表实现的,插入删除效率高,但查询效率低。
线程池的工作原理
- 线程池通过复用固定数量的线程来执行任务,减少了频繁创建和销毁线程的开销。
- 核心参数包括:核心线程数(corePoolSize)、最大线程数(maximumPoolSize)、队列(workQueue)、线程存活时间(keepAliveTime)等。
JUC中的ReentrantLock
- ReentrantLock是一个可重入的独占锁,支持公平锁和非公平锁的选择。
- 提供了
tryLock方法支持超时获取锁,lockInterruptibly方法支持中断获取锁。
Dubbo的核心架构
- Dubbo的核心架构包括:
- 注册中心:负责服务的注册与发现。
- 服务提供者:提供服务的具体实现。
- 服务消费者:调用服务提供者的接口。
- 通信协议:用于服务调用的通信。
- Dubbo的核心架构包括:
RabbitMQ的消息确认机制
- RabbitMQ的消息确认机制包括:
- 生产者确认(Publisher Confirm):确保消息成功发送到队列。
- 消费者确认(Consumer Acknowledgement):确保消息被成功消费。
- Consumer Acknowledgement分为自动确认和手动确认。
- RabbitMQ的消息确认机制包括: