互联网大厂最常见Java面试题汇总(附答案)

张开发
2026/4/21 17:12:37 15 分钟阅读
互联网大厂最常见Java面试题汇总(附答案)
全文分为十九个模块分别是 Java 基础、容器、多线程、反射、对象拷贝、Java Web 、异常、网络、设计模式、Spring/Spring MVC、Spring Boot/Spring Cloud、Hibernate、MyBatis、RabbitMQ、Kafka、Zookeeper、MySQL、Redis、JVM 如下图所示Java 基础1. JDK 和 JRE 有什么区别JDKJava Development Kit 的简称Java 开发工具包提供了 Java 的开发环境和运行环境。JREJava Runtime Environment 的简称Java 运行环境为 Java 的运行提供了所需环境。具体来说 JDK 其实包含了 JRE同时还包含了编译 Java 源码的编译器 Javac还包含了很多 Java 程序调试和分析的工具。简单来说如果你需要运行 Java 程序只需安装 JRE 就可以了如果你需要编写 Java 程序需要安装 JDK。2. 和 equals 的区别是什么 解读对于基本类型和引用类型 的作用效果是不同的如下所示基本类型比较的是值是否相同引用类型比较的是引用是否相同代码示例String x string; String y string; String z new String(string); System.out.println(xy); // true System.out.println(xz); // false System.out.println(x.equals(y)); // true System.out.println(x.equals(z)); // true代码解读因为 x 和 y 指向的是同一个引用所以 也是 true而 new String()方法则重写开辟了内存空间所以 结果为 false而 equals 比较的一直是值所以结果都为 true。equals 解读equals 本质上就是 只不过 String 和 Integer 等重写了 equals 方法把它变成了值比较。看下面的代码就明白了。首先来看默认情况下 equals 比较一个有相同值的对象代码如下class Cat { public Cat(String name) { this.name name; } private String name; public String getName() { return name; } public void setName(String name) { this.name name; } } Cat c1 new Cat(王磊); Cat c2 new Cat(王磊); System.out.println(c1.equals(c2)); // false输出结果出乎我们的意料竟然是 false这是怎么回事看了 equals 源码就知道了源码如下public boolean equals(Object obj) { return (this obj); }原来 equals 本质上就是 。那问题来了两个相同值的 String 对象为什么返回的是 true代码如下String s1 new String(老王); String s2 new String(老王); System.out.println(s1.equals(s2)); // true同样的当我们进入 String 的 equals 方法找到了答案代码如下public boolean equals(Object anObject) { if (this anObject) { return true; } if (anObject instanceof String) { String anotherString (String)anObject; int n value.length; if (n anotherString.value.length) { char v1[] value; char v2[] anotherString.value; int i 0; while (n-- ! 0) { if (v1[i] ! v2[i]) return false; i; } return true; } } return false; }原来是 String 重写了 Object 的 equals 方法把引用比较改成了值比较。总结 对于基本类型来说是值比较对于引用类型来说是比较的是引用而 equals 默认情况下是引用比较只是很多类重新了 equals 方法比如 String、Integer 等把它变成了值比较所以一般情况下 equals 比较的是值是否相等。3. 两个对象的 hashCode() 相同则 equals() 也一定为 true对吗不对两个对象的 hashCode() 相同equals() 不一定 true。代码示例String str1 通话; String str2 重地; System. out. println(String. format(str1%d | str2%d, str1\. hashCode(),str2\. hashCode())); System. out. println(str1\. equals(str2));执行的结果str11179395 | str21179395 false代码解读很显然“通话”和“重地”的 hashCode() 相同然而 equals() 则为 false因为在散列表中hashCode() 相等即两个键值对的哈希值相等然而哈希值相等并不一定能得出键值对相等。4. final 在 Java 中有什么作用final 修饰的类叫最终类该类不能被继承。final 修饰的方法不能被重写。final 修饰的变量叫常量常量必须初始化初始化之后值就不能被修改。5. Java 中的 Math. round(-1. 5) 等于多少等于 -1因为在数轴上取值时中间值0.5向右取整所以正 0.5 是往上取整负 0.5 是直接舍弃。6. String 属于基础的数据类型吗String 不属于基础类型基础类型有 8 种byte、boolean、char、short、int、float、long、double而 String 属于对象。7. Java 中操作字符串都有哪些类它们之间有什么区别操作字符串的类有String、StringBuffer、StringBuilder。String 和 StringBuffer、StringBuilder 的区别在于 String 声明的是不可变的对象每次操作都会生成新的 String 对象然后将指针指向新的 String 对象而 StringBuffer、StringBuilder 可以在原有对象的基础上进行操作所以在经常改变字符串内容的情况下最好不要使用 String。StringBuffer 和 StringBuilder 最大的区别在于StringBuffer 是线程安全的而 StringBuilder 是非线程安全的但 StringBuilder 的性能却高于 StringBuffer所以在单线程环境下推荐使用 StringBuilder多线程环境下推荐使用 StringBuffer。8. String stri与 String strnew String(i)一样吗不一样因为内存的分配方式不一样。String stri的方式Java 虚拟机会将其分配到常量池中而 String strnew String(i) 则会被分到堆内存中。9. 如何将字符串反转使用 StringBuilder 或者 stringBuffer 的 reverse() 方法。示例代码// StringBuffer reverse StringBuffer stringBuffer new StringBuffer(); stringBuffer. append(abcdefg); System. out. println(stringBuffer. reverse()); // gfedcba // StringBuilder reverse StringBuilder stringBuilder new StringBuilder(); stringBuilder. append(abcdefg); System. out. println(stringBuilder. reverse()); // gfedcba10. String 类的常用方法都有那些indexOf()返回指定字符的索引。charAt()返回指定索引处的字符。replace()字符串替换。trim()去除字符串两端空白。split()分割字符串返回一个分割后的字符串数组。getBytes()返回字符串的 byte 类型数组。length()返回字符串长度。toLowerCase()将字符串转成小写字母。toUpperCase()将字符串转成大写字符。substring()截取字符串。equals()字符串比较。11. 抽象类必须要有抽象方法吗不需要抽象类不一定非要有抽象方法。示例代码abstract class Cat { public static void sayHi() { System. out. println(hi~); } }上面代码抽象类并没有抽象方法但完全可以正常运行。12. 普通类和抽象类有哪些区别普通类不能包含抽象方法抽象类可以包含抽象方法。抽象类不能直接实例化普通类可以直接实例化。13. 抽象类能使用 final 修饰吗不能定义抽象类就是让其他类继承的如果定义为 final 该类就不能被继承这样彼此就会产生矛盾所以 final 不能修饰抽象类如下图所示编辑器也会提示错误信息14. 接口和抽象类有什么区别实现抽象类的子类使用 extends 来继承接口必须使用 implements 来实现接口。构造函数抽象类可以有构造函数接口不能有。实现数量类可以实现很多个接口但是只能继承一个抽象类。访问修饰符接口中的方法默认使用 public 修饰抽象类中的方法可以是任意访问修饰符。15. Java 中 IO 流分为几种按功能来分输入流input、输出流output。按类型来分字节流和字符流。字节流和字符流的区别是字节流按 8 位传输以字节为单位输入输出数据字符流按 16 位传输以字符为单位输入输出数据。16. BIO、NIO、AIO 有什么区别BIOBlock IO 同步阻塞式 IO就是我们平常使用的传统 IO它的特点是模式简单使用方便并发处理能力低。NIONon IO 同步非阻塞 IO是传统 IO 的升级客户端和服务器端通过 Channel通道通讯实现了多路复用。AIOAsynchronous IO 是 NIO 的升级也叫 NIO2实现了异步非堵塞 IO 异步 IO 的操作基于事件和回调机制。17. Files的常用方法都有哪些Files. exists()检测文件路径是否存在。Files. createFile()创建文件。Files. createDirectory()创建文件夹。Files. delete()删除一个文件或目录。Files. copy()复制文件。Files. move()移动文件。Files. size()查看文件个数。Files. read()读取文件。Files. write()写入文件。容器18. Java 容器都有哪些Java 容器分为 Collection 和 Map 两大类其下又有很多子类如下所示CollectionListArrayListLinkedListVectorStackSetHashSetLinkedHashSetTreeSetMapHashMapLinkedHashMapTreeMapConcurrentHashMapHashtable19. Collection 和 Collections 有什么区别Collection 是一个集合接口它提供了对集合对象进行基本操作的通用接口方法所有集合都是它的子类比如 List、Set 等。Collections 是一个包装类包含了很多静态方法不能被实例化就像一个工具类比如提供的排序方法 Collections. sort(list)。20. List、Set、Map 之间的区别是什么List、Set、Map 的区别主要体现在两个方面元素是否有序、是否允许元素重复。三者之间的区别如下表21. HashMap 和 Hashtable 有什么区别存储HashMap 运行 key 和 value 为 null而 Hashtable 不允许。线程安全Hashtable 是线程安全的而 HashMap 是非线程安全的。推荐使用在 Hashtable 的类注释可以看到Hashtable 是保留类不建议使用推荐在单线程环境下使用 HashMap 替代如果需要多线程使用则用 ConcurrentHashMap 替代。22. 如何决定使用 HashMap 还是 TreeMap对于在 Map 中插入、删除、定位一个元素这类操作HashMap 是最好的选择因为相对而言 HashMap 的插入会更快但如果你要对一个 key 集合进行有序的遍历那 TreeMap 是更好的选择。23. 说一下 HashMap 的实现原理HashMap 基于 Hash 算法实现的我们通过 put(key,value)存储get(key)来获取。当传入 key 时HashMap 会根据 key. hashCode() 计算出 hash 值根据 hash 值将 value 保存在 bucket 里。当计算出的 hash 值相同时我们称之为 hash 冲突HashMap 的做法是用链表和红黑树存储相同 hash 值的 value。当 hash 冲突的个数比较少时使用链表否则使用红黑树。24. 说一下 HashSet 的实现原理HashSet 是基于 HashMap 实现的HashSet 底层使用 HashMap 来保存所有元素因此 HashSet 的实现比较简单相关 HashSet 的操作基本上都是直接调用底层 HashMap 的相关方法来完成HashSet 不允许重复的值。25. ArrayList 和 LinkedList 的区别是什么数据结构实现ArrayList 是动态数组的数据结构实现而 LinkedList 是双向链表的数据结构实现。随机访问效率ArrayList 比 LinkedList 在随机访问的时候效率要高因为 LinkedList 是线性的数据存储方式所以需要移动指针从前往后依次查找。增加和删除效率在非首尾的增加和删除操作LinkedList 要比 ArrayList 效率要高因为 ArrayList 增删操作要影响数组内的其他数据的下标。综合来说在需要频繁读取集合中的元素时更推荐使用 ArrayList而在插入和删除操作较多时更推荐使用 LinkedList。26. 如何实现数组和 List 之间的转换数组转 List使用 Arrays. asList(array) 进行转换。List 转数组使用 List 自带的 toArray() 方法。代码示例// list to array ListString list new ArrayListString(); list. add(王磊); list. add(的博客); list. toArray(); // array to list String[] array new String[]{王磊,的博客}; Arrays. asList(array);27. ArrayList 和 Vector 的区别是什么线程安全Vector 使用了 Synchronized 来实现线程同步是线程安全的而 ArrayList 是非线程安全的。性能ArrayList 在性能方面要优于 Vector。扩容ArrayList 和 Vector 都会根据实际的需要动态的调整容量只不过在 Vector 扩容每次会增加 1 倍而 ArrayList 只会增加 50%。28. Array 和 ArrayList 有何区别Array 可以存储基本数据类型和对象ArrayList 只能存储对象。Array 是指定固定大小的而 ArrayList 大小是自动扩展的。Array 内置方法没有 ArrayList 多比如 addAll、removeAll、iteration 等方法只有 ArrayList 有。29. 在 Queue 中 poll()和 remove()有什么区别相同点都是返回第一个元素并在队列中删除返回的对象。不同点如果没有元素 poll()会返回 null而 remove()会直接抛出 NoSuchElementException 异常。代码示例QueueString queue new LinkedListString(); queue. offer(string); // add System. out. println(queue. poll()); System. out. println(queue. remove()); System. out. println(queue. size());30. 哪些集合类是线程安全的Vector、Hashtable、Stack 都是线程安全的而像 HashMap 则是非线程安全的不过在 JDK 1.5 之后随着 Java. util. concurrent 并发包的出现它们也有了自己对应的线程安全类比如 HashMap 对应的线程安全类就是 ConcurrentHashMap。31. 迭代器 Iterator 是什么Iterator 接口提供遍历任何 Collection 的接口。我们可以从一个 Collection 中使用迭代器方法来获取迭代器实例。迭代器取代了 Java 集合框架中的 Enumeration迭代器允许调用者在迭代过程中移除元素。32. Iterator 怎么使用有什么特点Iterator 使用代码如下ListString list new ArrayList(); IteratorString it list. iterator(); while(it. hasNext()){ String obj it. next(); System. out. println(obj); }Iterator 的特点是更加安全因为它可以确保在当前遍历的集合元素被更改的时候就会抛出 ConcurrentModificationException 异常。33. Iterator 和 ListIterator 有什么区别Iterator 可以遍历 Set 和 List 集合而 ListIterator 只能遍历 List。Iterator 只能单向遍历而 ListIterator 可以双向遍历向前/后遍历。ListIterator 从 Iterator 接口继承然后添加了一些额外的功能比如添加一个元素、替换一个元素、获取前面或后面元素的索引位置。34. 怎么确保一个集合不能被修改可以使用 Collections. unmodifiableCollection(Collection c) 方法来创建一个只读集合这样改变集合的任何操作都会抛出 Java. lang. UnsupportedOperationException 异常。示例代码如下ListString list new ArrayList(); list. add(x); CollectionString clist Collections. unmodifiableCollection(list); clist. add(y); // 运行时此行报错 System. out. println(list. size());多线程35. 并行和并发有什么区别并行多个处理器或多核处理器同时处理多个任务。并发多个任务在同一个 CPU 核上按细分的时间片轮流(交替)执行从逻辑上来看那些任务是同时执行。如下图并发 两个队列和一台咖啡机。并行 两个队列和两台咖啡机。36. 线程和进程的区别一个程序下至少有一个进程一个进程下至少有一个线程一个进程下也可以有多个线程来增加程序的执行速度。37. 守护线程是什么守护线程是运行在后台的一种特殊进程。它独立于控制终端并且周期性地执行某种任务或等待处理某些发生的事件。在 Java 中垃圾回收线程就是特殊的守护线程。38. 创建线程有哪几种方式创建线程有三种方式继承 Thread 重写 run 方法实现 Runnable 接口实现 Callable 接口。39. 说一下 runnable 和 callable 有什么区别runnable 没有返回值callable 可以拿到有返回值callable 可以看作是 runnable 的补充。40. 线程有哪些状态线程的状态NEW 尚未启动RUNNABLE 正在执行中BLOCKED 阻塞的被同步锁或者IO锁阻塞WAITING 永久等待状态TIMED_WAITING 等待指定的时间重新被唤醒的状态TERMINATED 执行完成41. sleep() 和 wait() 有什么区别类的不同sleep() 来自 Threadwait() 来自 Object。释放锁sleep() 不释放锁wait() 释放锁。用法不同sleep() 时间到会自动恢复wait() 可以使用 notify()/notifyAll()直接唤醒。42. notify()和 notifyAll()有什么区别notifyAll()会唤醒所有的线程notify()之后唤醒一个线程。notifyAll() 调用后会将全部线程由等待池移到锁池然后参与锁的竞争竞争成功则继续执行如果不成功则留在锁池等待锁被释放后再次参与竞争。而 notify()只会唤醒一个线程具体唤醒哪一个线程由虚拟机控制。43. 线程的 run() 和 start() 有什么区别start() 方法用于启动线程run() 方法用于执行线程的运行时代码。run() 可以重复调用而 start() 只能调用一次。44. 创建线程池有哪几种方式线程池创建有七种方式最核心的是最后一种newSingleThreadExecutor()它的特点在于工作线程数目被限制为 1操作一个无界的工作队列所以它保证了所有任务的都是被顺序执行最多会有一个任务处于活动状态并且不允许使用者改动线程池实例因此可以避免其改变线程数目newCachedThreadPool()它是一种用来处理大量短时间工作任务的线程池具有几个鲜明特点它会试图缓存线程并重用当无缓存线程可用时就会创建新的工作线程如果线程闲置的时间超过 60 秒则被终止并移出缓存长时间闲置时这种线程池不会消耗什么资源。其内部使用 SynchronousQueue 作为工作队列newFixedThreadPool(int nThreads)重用指定数目nThreads的线程其背后使用的是无界的工作队列任何时候最多有 nThreads 个工作线程是活动的。这意味着如果任务数量超过了活动队列数目将在工作队列中等待空闲线程出现如果有工作线程退出将会有新的工作线程被创建以补足指定的数目 nThreadsnewSingleThreadScheduledExecutor()创建单线程池返回 ScheduledExecutorService可以进行定时或周期性的工作调度newScheduledThreadPool(int corePoolSize)和newSingleThreadScheduledExecutor()类似创建的是个 ScheduledExecutorService可以进行定时或周期性的工作调度区别在于单一工作线程还是多个工作线程newWorkStealingPool(int parallelism)这是一个经常被人忽略的线程池Java 8 才加入这个创建方法其内部会构建ForkJoinPool利用Work-Stealing算法并行地处理任务不保证处理顺序ThreadPoolExecutor()是最原始的线程池创建上面1-3创建方式都是对ThreadPoolExecutor的封装。45. 线程池都有哪些状态RUNNING这是最正常的状态接受新的任务处理等待队列中的任务。SHUTDOWN不接受新的任务提交但是会继续处理等待队列中的任务。STOP不接受新的任务提交不再处理等待队列中的任务中断正在执行任务的线程。TIDYING所有的任务都销毁了workCount 为 0线程池的状态在转换为 TIDYING 状态时会执行钩子方法 terminated()。TERMINATEDterminated()方法结束后线程池的状态就会变成这个。46. 线程池中 submit() 和 execute() 方法有什么区别execute()只能执行 Runnable 类型的任务。submit()可以执行 Runnable 和 Callable 类型的任务。Callable 类型的任务可以获取执行的返回值而 Runnable 执行无返回值。47. 在 Java 程序中怎么保证多线程的运行安全方法一使用安全类比如 Java. util. concurrent 下的类。方法二使用自动锁 synchronized。方法三使用手动锁 Lock。手动锁 Java 示例代码如下Lock lock new ReentrantLock(); lock. lock(); try { System. out. println(获得锁); } catch (Exception e) { // TODO: handle exception } finally { System. out. println(释放锁); lock. unlock(); }48. 多线程中 synchronized 锁升级的原理是什么synchronized 锁升级原理在锁对象的对象头里面有一个 threadid 字段在第一次访问的时候 threadid 为空jvm 让其持有偏向锁并将 threadid 设置为其线程 id再次进入的时候会先判断 threadid 是否与其线程 id 一致如果一致则可以直接使用此对象如果不一致则升级偏向锁为轻量级锁通过自旋循环一定次数来获取锁执行一定次数之后如果还没有正常获取到要使用的对象此时就会把锁从轻量级升级为重量级锁此过程就构成了 synchronized 锁的升级。锁的升级的目的锁升级是为了减低了锁带来的性能消耗。在 Java 6 之后优化 synchronized 的实现方式使用了偏向锁升级为轻量级锁再升级到重量级锁的方式从而减低了锁带来的性能消耗。49. 什么是死锁当线程 A 持有独占锁a并尝试去获取独占锁 b 的同时线程 B 持有独占锁 b并尝试获取独占锁 a 的情况下就会发生 AB 两个线程由于互相持有对方需要的锁而发生的阻塞现象我们称为死锁。50. 怎么防止死锁尽量使用 tryLock(long timeout, TimeUnit unit)的方法(ReentrantLock、ReentrantReadWriteLock)设置超时时间超时可以退出防止死锁。尽量使用 Java. util. concurrent 并发类代替自己手写锁。尽量降低锁的使用粒度尽量不要几个功能用同一把锁。尽量减少同步的代码块。51. ThreadLocal 是什么有哪些使用场景ThreadLocal 为每个使用该变量的线程提供独立的变量副本所以每一个线程都可以独立地改变自己的副本而不会影响其它线程所对应的副本。ThreadLocal 的经典使用场景是数据库连接和 session 管理等。52. 说一下 synchronized 底层实现原理synchronized 是由一对 monitorenter/monitorexit 指令实现的monitor 对象是同步的基本实现单元。在 Java 6 之前monitor 的实现完全是依靠操作系统内部的互斥锁因为需要进行用户态到内核态的切换所以同步操作是一个无差别的重量级操作性能也很低。但在 Java 6 的时候Java 虚拟机 对此进行了大刀阔斧地改进提供了三种不同的 monitor 实现也就是常说的三种不同的锁偏向锁Biased Locking、轻量级锁和重量级锁大大改进了其性能。53. synchronized 和 volatile 的区别是什么volatile 是变量修饰符synchronized 是修饰类、方法、代码段。volatile 仅能实现变量的修改可见性不能保证原子性而 synchronized 则可以保证变量的修改可见性和原子性。volatile 不会造成线程的阻塞synchronized 可能会造成线程的阻塞。54. synchronized 和 Lock 有什么区别synchronized 可以给类、方法、代码块加锁而 lock 只能给代码块加锁。synchronized 不需要手动获取锁和释放锁使用简单发生异常会自动释放锁不会造成死锁而 lock 需要自己加锁和释放锁如果使用不当没有 unLock()去释放锁就会造成死锁。通过 Lock 可以知道有没有成功获取锁而 synchronized 却无法办到。55. synchronized 和 ReentrantLock 区别是什么synchronized 早期的实现比较低效对比 ReentrantLock大多数场景性能都相差较大但是在 Java 6 中对 synchronized 进行了非常多的改进。主要区别如下ReentrantLock 使用起来比较灵活但是必须有释放锁的配合动作ReentrantLock 必须手动获取与释放锁而 synchronized 不需要手动释放和开启锁ReentrantLock 只适用于代码块锁而 synchronized 可用于修饰方法、代码块等。volatile 标记的变量不会被编译器优化synchronized 标记的变量可以被编译器优化。56. 说一下 atomic 的原理atomic 主要利用 CAS (Compare And Wwap) 和 volatile 和 native 方法来保证原子操作从而避免 synchronized 的高开销执行效率大为提升。反射57. 什么是反射反射是在运行状态中对于任意一个类都能够知道这个类的所有属性和方法对于任意一个对象都能够调用它的任意一个方法和属性这种动态获取的信息以及动态调用对象的方法的功能称为 Java 语言的反射机制。58. 什么是 Java 序列化什么情况下需要序列化Java 序列化是为了保存各种对象在内存中的状态并且可以把保存的对象状态再读出来。以下情况需要使用 Java 序列化想把的内存中的对象状态保存到一个文件中或者数据库中时候想用套接字在网络上传送对象的时候想通过RMI远程方法调用传输对象的时候。59. 动态代理是什么有哪些应用动态代理是运行时动态生成代理类。动态代理的应用有 spring aop、hibernate 数据查询、测试框架的后端 mock、rpcJava注解对象获取等。60. 怎么实现动态代理JDK 原生动态代理和 cglib 动态代理。JDK 原生动态代理是基于接口实现的而 cglib 是基于继承当前类的子类实现的。对象拷贝61. 为什么要使用克隆克隆的对象可能包含一些已经修改过的属性而 new 出来的对象的属性都还是初始化时候的值所以当需要一个新的对象来保存当前对象的“状态”就靠克隆方法了。62. 如何实现对象克隆实现 Cloneable 接口并重写 Object 类中的 clone() 方法。实现 Serializable 接口通过对象的序列化和反序列化实现克隆可以实现真正的深度克隆。63. 深拷贝和浅拷贝区别是什么浅克隆当对象被复制时只复制它本身和其中包含的值类型的成员变量而引用类型的成员对象并没有复制。深克隆除了对象本身被复制外对象所包含的所有成员变量也将复制。由于字数限制无法全篇展示需要全内容的小伙伴可以点击下方名片获取

更多文章