大厂常见问题
通用问题
参考:
https://blog.csdn.net/u012456528/article/details/79759487
Java 面试随着时间的改变而改变。在过去的日子里,当你知道 String 和 StringBuilder 的区别(String 类型和 StringBuffer 类型的主要性能区别其实在于 String 是不可变的对象。因此在每次对 String 类型进行改变的时候其实都等同于生成了一个新的 String 对象,然后将指针指向新的 String 对象,所以经常改变内容的字符串最好不要用 String ,因为每次生成对象都会对系统性能产生影响,特别当内存中无引用对象多了以后,JVM 的 GC 就会开始工作,影响性能,可以考虑使用可变字符序列StringBuilder)就能让你直接进入第二轮面试,但是现在问题变得越来越高级,面试官问的问题也更深入。 在我初入职场的时候,类似于 Vector 与 Array 的区别、HashMap 与 Hashtable 的区别是最流行的问题,只需要记住它们,就能在面试中获得更好的机会,但这种情形已经不复存在。如今,你将会被问到许多 Java 程序员都没有看过的领域,如 NIO,设计模式,成熟的单元测试,或者那些很难掌握的知识,如并发、算法、数据结构及编码。
由于我喜欢研究面试题,因此我已经收集了许多的面试问题,包括许多许多不同的主题。我已经为这众多的问题准备一段时间了,现在我将它们分享给你们。这里面不但包含经典的面试问题,如线程、集合、equals 和 hashcode、socket,而且还包含了 NIO、数组、字符串、Java 8 等主题。
该列表包含了入门级 Java 程序员和多年经验的高级开发者的问题。无论你是 1、2、3、4、5、6、7、8、9 还是 10 年经验的开发者,你都能在其中找到一些有趣的问题。这里包含了一些超级容易回答的问题,同时包含经验丰富的 Java 程序员也会棘手的问题。
当然你们也是非常幸运的,当今有许多好的书来帮助你准备 Java 面试,其中有一本我觉得特别有用和有趣的是 Markham 的 Java 程序面试揭秘(Java Programming Interview Exposed)。 这本书会告诉你一些 Java 和 JEE 面试中最重要的主题,即使你不是准备 Java 面试,也值得一读。
该问题列表特别长,我们有各个地方的问题,所以,答案必须要短小、简洁、干脆,不拖泥带水。因此,除了这一个段落,你只会听到问题与答案,再无其他内容,没有反馈,也没有评价。为此,我已经写好了一些博文,在这些文章中你可以找到我对某些问题的观点,如我为什么喜欢这个问题,这个问题的挑战是什么?期望从面试者那获取到什么样的答案?
这个列表有一点不同,我鼓励你采用类似的方式去分享问题和答案,这样容易温习。我希望这个列表对面试官和候选人都有很好的用处,面试官可以对这些问题上做一些改变以获取新奇和令人惊奇的元素,这对一次好的面试来说非常重要。而候选者,可以扩展和测试 Java 程序语言和平台关键领域的知识。2015 年,会更多的关注并发概念,JVM 内部,32 位 JVM 和 64 JVM的区别,单元测试及整洁的代码。我确信,如果你读过这个庞大的 Java 面试问题列表,无论是电话面试还是面对面的面试,你都能有很好的表现。
Java 面试中的重要话题
除了你看到的惊人的问题数量,我也尽量保证质量。我不止一次分享各个重要主题中的问题,也确保包含所谓的高级话题,这些话题很多程序员不喜欢准备或者直接放弃,因为他们的工作不会涉及到这些。Java NIO 和 JVM 底层就是最好的例子。你也可以将设计模式划分到这一类中,但是越来越多有经验的程序员了解 GOF 设计模式并应用这些模式。我也尽量在这个列表中包含 2015 年最新的面试问题,这些问题可能是来年关注的核心。为了给你一个大致的了解,下面列出这份 Java 面试问题列表包含的主题:
120 大 Java 面试题及答案
现在是时候给你展示我近 5 年从各种面试中收集来的 120 个问题了。我确定你在自己的面试中见过很多这些问题,很多问题你也能正确回答。
多线程、并发及线程的基础问题
1)Java 中能创建 volatile 数组吗?
能,Java 中可以创建 volatile 类型数组,不过只是一个指向数组的引用,而不是整个数组。我的意思是,如果改变引用指向的数组,将会受到 volatile 的保护,但是如果多个线程同时改变数组的元素,volatile 标示符就不能起到之前的保护作用了。
2)volatile 能使得一个非原子操作变成原子操作吗?
一个典型的例子是在类中有一个 long 类型的成员变量。如果你知道该成员变量会被多个线程访问,如计数器、价格等,你最好是将其设置为 volatile。为什么?因为 Java 中读取 long 类型变量不是原子的,需要分成两步,如果一个线程正在修改该 long 变量的值,另一个线程可能只能看到该值的一半(前 32 位)。但是对一个 volatile 型的 long 或 double 变量的读写是原子。
3)volatile 修饰符的有过什么实践?
一种实践是用 volatile 修饰 long 和 double 变量,使其能按原子类型来读写。double 和 long 都是64位宽,因此对这两种类型的读是分为两部分的,第一次读取第一个 32 位,然后再读剩下的 32 位,这个过程不是原子的,但 Java 中 volatile 型的 long 或 double 变量的读写是原子的。volatile 修复符的另一个作用是提供内存屏障(memory barrier),例如在分布式框架中的应用。简单的说,就是当你写一个 volatile 变量之前,Java 内存模型会插入一个写屏障(write barrier),读一个 volatile 变量之前,会插入一个读屏障(read barrier)。意思就是说,在你写一个 volatile 域时,能保证任何线程都能看到你写的值,同时,在写之前,也能保证任何数值的更新对所有线程是可见的,因为内存屏障会将其他所有写的值更新到缓存。
4)volatile 类型变量提供什么保证?
volatile 变量提供顺序和可见性保证,例如,JVM 或者 JIT为了获得更好的性能会对语句重排序,但是 volatile 类型变量即使在没有同步块的情况下赋值也不会与其他语句重排序。 volatile 提供 happens-before 的保证,确保一个线程的修改能对其他线程是可见的。某些情况下,volatile 还能提供原子性,如读 64 位数据类型,像 long 和 double 都不是原子的,但 volatile 类型的 double 和 long 就是原子的。
5) 10 个线程和 2 个线程的同步代码,哪个更容易写?
从写代码的角度来说,两者的复杂度是相同的,因为同步代码与线程数量是相互独立的。但是同步策略的选择依赖于线程的数量,因为越多的线程意味着更大的竞争,所以你需要利用同步技术,如锁分离,这要求更复杂的代码和专业知识。
6)你是如何调用 wait()方法的?使用 if 块还是循环?为什么?
wait() 方法应该在循环调用,因为当线程获取到 CPU 开始执行的时候,其他条件可能还没有满足,所以在处理前,循环检测条件是否满足会更好。
下面是一段标准的使用 wait 和 notify 方法的代码:
// The standard idiom for using the wait method
synchronized (obj) { while (condition does not hold) obj.wait();
// (Releases lock, and reacquires on wakeup) ... // Perform action appropriate to condition }
参见 Effective Java 第 69 条,获取更多关于为什么应该在循环中来调用 wait 方法的内容。
7)什么是多线程环境下的伪共享(false sharing)?
伪共享是多线程系统(每个处理器有自己的局部缓存)中一个众所周知的性能问题。伪共享发生在不同处理器的上的线程对变量的修改依赖于相同的缓存行,伪共享问题很难被发现,因为线程可能访问完全不同的全局变量,内存中却碰巧在很相近的位置上。如其他诸多的并发问题,避免伪共享的最基本方式是仔细审查代码,根据缓存行来调整你的数据结构。
有经验程序员的 Java 面试题
8)什么是 Busy spin?我们为什么要使用它?
Busy spin 是一种在不释放 CPU 的基础上等待事件的技术。它经常用于避免丢失 CPU 缓存中的数据(如果线程先暂停,之后在其他CPU上运行就会丢失)。所以,如果你的工作要求低延迟,并且你的线程目前没有任何顺序,这样你就可以通过循环检测队列中的新消息来代替调用 sleep() 或 wait() 方法。它唯一的好处就是你只需等待很短的时间,如几微秒或几纳秒。LMAX 分布式框架是一个高性能线程间通信的库,该库有一个 BusySpinWaitStrategy 类就是基于这个概念实现的,使用 busy spin 循环 EventProcessors 等待屏障。
9)Java 中怎么获取一份线程 dump 文件?
在 Linux 下,你可以通过命令 kill -3 PID (Java 进程的进程 ID)来获取 Java 应用的 dump 文件。在 Windows 下,你可以按下 Ctrl + Break 来获取。这样 JVM 就会将线程的 dump 文件打印到标准输出或错误文件中,它可能打印在控制台或者日志文件中,具体位置依赖应用的配置。如果你使用Tomcat。
10)Swing 是线程安全的?
不是,Swing 不是线程安全的。你不能通过任何线程来更新 Swing 组件,如 JTable、JList 或 JPanel,事实上,它们只能通过 GUI 或 AWT 线程来更新。这就是为什么 Swing 提供 invokeAndWait() 和 invokeLater() 方法来获取其他线程的 GUI 更新请求。这些方法将更新请求放入 AWT 的线程队列中,可以一直等待,也可以通过异步更新直接返回结果。你也可以在参考答案中查看和学习到更详细的内容。
11)什么是线程局部变量?
当使用ThreadLocal维护变量时,ThreadLocal为每个使用该变量的线程提供独立的变量副本,每个线程都可以独立地改变自己的副本,而不会影响其它线程所对应的副本,是线程隔离的。线程隔离的秘密在于ThreadLocalMap类(ThreadLocal的静态内部类)
线程局部变量是局限于线程内部的变量,属于线程自身所有,不在多个线程间共享。Java 提供 ThreadLocal 类来支持线程局部变量,是一种实现线程安全的方式。但是在管理环境下(如 web 服务器)使用线程局部变量的时候要特别小心,在这种情况下,工作线程的生命周期比任何应用变量的生命周期都要长。任何线程局部变量一旦在工作完成后没有释放,Java 应用就存在内存泄露的风险。
ThreadLocal的方法:void set(T value)、T get()以及T initialValue()。
ThreadLocal是如何为每个线程创建变量的副本的:
首先,在每个线程Thread内部有一个ThreadLocal.ThreadLocalMap类型的成员变量threadLocals,这个threadLocals就是用来存储实际的变量副本的,键值为当前ThreadLocal变量,value为变量副本(即T类型的变量)。初始时,在Thread里面,threadLocals为空,当通过ThreadLocal变量调用get()方法或者set()方法,就会对Thread类中的threadLocals进行初始化,并且以当前ThreadLocal变量为键值,以ThreadLocal要保存的副本变量为value,存到threadLocals。然后在当前线程里面,如果要使用副本变量,就可以通过get方法在threadLocals里面查找。
总结:
a、实际的通过ThreadLocal创建的副本是存储在每个线程自己的threadLocals中的
b、为何threadLocals的类型ThreadLocalMap的键值为ThreadLocal对象,因为每个线程中可有多个threadLocal变量,就像上面代码中的longLocal和stringLocal;
c、在进行get之前,必须先set,否则会报空指针异常;如果想在get之前不需要调用set就能正常访问的话,必须重写initialValue()方法
12)用 wait-notify 写一段代码来解决生产者-消费者问题?
13) 用 Java 写一个线程安全的单例模式(Singleton)?
14)Java 中 sleep 方法和 wait 方法的区别?
15)什么是不可变对象(immutable object)?Java 中怎么创建一个不可变对象?
16)我们能创建一个包含可变对象的不可变对象吗?
17)Java 中应该使用什么数据类型来代表价格?
18)怎么将 byte 转换为 String?
19)Java 中怎样将 bytes 转换为 long 类型?
20)我们能将 int 强制转换为 byte 类型的变量吗?如果该值大于 byte 类型的范围,将会出现什么现象?
21)存在两个类,B 继承 A,C 继承 B,我们能将 B 转换为 C 么?如 C = (C) B;
22)哪个类包含 clone 方法?是 Cloneable 还是 Object?
23)Java 中 ++ 操作符是线程安全的吗?
24)a = a + b 与 a += b 的区别
25)我能在不进行强制转换的情况下将一个 double 值赋值给 long 类型的变量吗?
26)3*0.1 == 0.3 将会返回什么?true 还是 false?
28)为什么 Java 中的 String 是不可变的(Immutable)?
29)我们能在 Switch 中使用 String 吗?
30)Java 中的构造器链是什么?
JVM 底层 与 GC(Garbage Collection) 的面试问题
31)64 位 JVM 中,int 的长度是多数?
32)Serial 与 Parallel GC之间的不同之处?
33)32 位和 64 位的 JVM,int 类型变量的长度是多数?
34)Java 中 WeakReference 与 SoftReference的区别?
35)WeakHashMap 是怎么工作的?
36)JVM 选项 -XX:+UseCompressedOops 有什么作用?为什么要使用?
37)怎样通过 Java 程序来判断 JVM 是 32 位 还是 64 位?
38)32 位 JVM 和 64 位 JVM 的最大堆内存分别是多数?
39)JRE、JDK、JVM 及 JIT 之间有什么不同?
3 年工作经验的 Java 面试题
40)解释 Java 堆空间及 GC?
JVM 底层面试题及答案
41)你能保证 GC 执行吗?
42)怎么获取 Java 程序使用的内存?堆使用的百分比?
43)Java 中堆和栈有什么区别?
Java 基本概念面试题
44)“a==b”和”a.equals(b)”有什么区别?
45)a.hashCode() 有什么用?与 a.equals(b) 有什么关系?
46)final、finalize 和 finally 的不同之处?
47)Java 中的编译期常量是什么?使用它又什么风险?
Java 集合框架的面试题
48) List、Set、Map 和 Queue 之间的区别
49)poll() 方法和 remove() 方法的区别?
50)Java 中 LinkedHashMap 和 PriorityQueue 的区别是什么?
51)ArrayList 与 LinkedList 的不区别?
52)用哪两种方式来实现集合的排序?
53)Java 中怎么打印数组?
54)Java 中的 LinkedList 是单向链表还是双向链表?
55)Java 中的 TreeMap 是采用什么树实现的?
56) Hashtable 与 HashMap 有什么不同之处?
57)Java 中的 HashSet,内部是如何工作的?(answer答案)
58)写一段代码在遍历 ArrayList 时移除一个元素?(答案)
59)我们能自己写一个容器类,然后使用 for-each 循环吗?
60)ArrayList 和 HashMap 的默认大小是多数?(答案)
61)有没有可能两个不相等的对象有有相同的 hashcode?
62)两个相同的对象会有不同的的 hash code 吗?
63)我们可以在 hashcode() 中使用随机数字吗?(答案)
64)Java 中,Comparator 与 Comparable 有什么不同?(答案)
65)为什么在重写 equals 方法的时候需要重写 hashCode 方法?(答案)
Java IO 和 NIO 的面试题
66)在我 Java 程序中,我有三个 socket,我需要多少个线程来处理?
67)Java 中怎么创建 ByteBuffer?
68)Java 中,怎么读写 ByteBuffer ?
69)Java 采用的是大端还是小端?
70)ByteBuffer 中的字节序是什么?
71)Java 中,直接缓冲区与非直接缓冲器有什么区别?
72)Java 中的内存映射缓存区是什么?
73)socket 选项 TCP NO DELAY 是指什么?
74)TCP 协议与 UDP 协议有什么区别?
75)Java 中,ByteBuffer 与 StringBuffer有什么区别?
Java 最佳实践的面试问题
76)Java 中,编写多线程程序的时候你会遵循哪些最佳实践?(答案)
78)说出至少 5 点在 Java 中使用线程的最佳实践。(答案)
79)说出 5 条 IO 的最佳实践(答案)
80)列出 5 个应该遵循的 JDBC 最佳实践(答案)
81)说出几条 Java 中方法重载的最佳实践?(答案)
Date、Time 及 Calendar 的面试题
82)在多线程环境下,SimpleDateFormat 是线程安全的吗?(答案)
83)Java 中如何格式化一个日期?如格式化为 ddMMyyyy 的形式?(答案)
84)Java 中,怎么在格式化的日期中显示时区?
85)Java 中 java.util.Date 与 java.sql.Date 有什么区别?
86)Java 中,如何计算两个日期之间的差距?
87)Java 中,如何将字符串 YYYYMMDD 转换为日期?
单元测试 JUnit 面试题
89)如何测试静态方法?
90)怎么利用 JUnit 来测试一个方法的异常?
91)你使用过哪个单元测试库来测试你的 Java 程序?
92)@Before 和 @BeforeClass 有什么区别?
编程和代码相关的面试题
93)怎么检查一个字符串只包含数字?
94)Java 中如何利用泛型写一个 LRU 缓存?
95)写一段 Java 程序将 byte 转换为 long?
96)在不使用 StringBuffer 的前提下,怎么反转一个字符串?
97)Java 中,怎么获取一个文件中单词出现的最高频率?
98)如何检查出两个给定的字符串是反序的?
99)Java 中,怎么打印出一个字符串的所有排列?
100)Java 中,怎样才能打印出数组中的重复元素?
101)Java 中如何将字符串转换为整数?
102)在没有使用临时变量的情况如何交换两个整数变量的值?
关于 OOP 和设计模式的面试题
103)接口是什么?为什么要使用接口而不是直接使用具体类?
104)Java 中,抽象类与接口之间有什么不同?(答案)
105)除了单例模式,你在生产环境中还用过什么设计模式?
106)你能解释一下里氏替换原则吗?
107) 什么情况下会违反迪米特法则?为什么会有这个问题?(答案)
108)适配器模式是什么?什么时候使用?
109)什么是“依赖注入”和“控制反转”?为什么有人使用?
110)抽象类是什么?它与接口有什么区别?你为什么要使用过抽象类?(答案)
111)构造器注入和 setter 依赖注入,那种方式更好?(答案)
112)依赖注入和工厂模式之间有什么不同?(答案)
113)适配器模式和装饰器模式有什么区别?(答案)
114)适配器模式和代理模式之前有什么不同?(答案)
115)什么是模板方法模式?(答案)
116)什么时候使用访问者模式?(答案)
117)什么时候使用组合模式?(答案)
118)继承和组合之间有什么不同?(答案)
119)描述 Java 中的重载和重写?(答案)
120)Java 中,嵌套公共静态类与顶级类有什么不同?(答案)
121) OOP 中的 组合、聚合和关联有什么区别?(答案)
122)给我一个符合开闭原则的设计模式的例子?(答案)
123)抽象工厂模式和原型模式之间的区别?
124)什么时候使用享元模式?
Java 面试中其他各式各样的问题
125)嵌套静态类与顶级类有什么区别?(答案)
126)你能写出一个正则表达式来判断一个字符串是否是一个数字吗?(解决方案)
127)Java 中,受检查异常 和 不受检查异常的区别?(答案)
128)Java 中,throw 和 throws 有什么区别?(答案)
129)Java 中,Serializable 与 Externalizable 的区别?(答案)
130)Java 中,DOM 和 SAX 解析器有什么不同?(答案)
131)说出 JDK 1.7 中的三个新特性?(答案)
132)说出 5 个 JDK 1.8 引入的新特性?(答案)
133)Java 中,Maven 和 ANT 有什么区别?(答案)
由于答案解析部分太长了,给大家总结到文档里面了,还有楼主收藏的一些很不错的面试资料,如果你知道这些问题的答案,欢迎在评论出答出来,祝大家面试成功~添加小助手V:maxiaoananan即可获取