Java面试题 01
目录
5. HashMap 底层是 数组 + 链表 + 红黑树,为什么要用这几类结构?
1️⃣ TLAB(Thread Local Allocation Buffer)
10. Java 语言中关键字 static 的作用是什么?
1. Java 中不可变对象的优缺点?使用场景有哪些?
✅ 优点:
- 线程安全:无需同步,天然支持多线程共享。
- 缓存友好:可被缓存(如 String 常量池),提升性能。
- 简化编程:状态不会改变,逻辑更清晰,减少 bug。
- 可作为 Map 的 Key:因为 hashCode 和 equals 不会变。
❌ 缺点:
- 创建新对象开销大:每次“修改”都要新建对象,可能增加 GC 压力。
- 不适合频繁变更状态的场景:比如计数器、缓冲区等。
🎯 使用场景:
String、Integer等包装类- 配置项、常量、枚举
- 函数式编程中的纯函数返回值
- 多线程共享数据(如 ConcurrentHashMap 的 key)
💡 示例:
String s = "hello"; s += " world";实际是创建了新的 String 对象。
2. Java 中数组是对象吗?
✅ 是的,数组是对象!
- 数组在 JVM 中是以对象形式存在的,继承自
Object类。 - 可以调用
toString()、hashCode()、getClass()等方法。 - 有
length属性(不是方法)。 - 可以通过反射操作数组。
int[] arr = {1, 2, 3};
System.out.println(arr.getClass()); // 输出: [I
System.out.println(arr instanceof Object); // true
⚠️ 注意:基本类型数组(如
int[])和引用类型数组(如String[])都是对象,但它们的内部存储不同。
3. 什么是值传递和引用传递?
Java 只有值传递,没有真正的“引用传递”。
🔹 值传递:
- 传递的是变量的副本。
- 对于基本类型:复制的是数值。
- 对于引用类型:复制的是引用的地址值(即指向对象的指针)。
举例:
public void changeValue(int x) { x = 10; } // 不影响原变量
public void changeRef(StringBuilder sb) {
sb.append("world"); // 修改对象内容 → 影响原对象
sb = new StringBuilder(); // 重新赋值引用 → 不影响原引用
}
✅ 结论:Java 中所有参数传递都是“按值传递”,只是对于对象,传的是“引用的值”。
4. 设计接口时要注意什么?
✅ 核心原则:
- 单一职责:一个接口只定义一类行为。
- 命名规范:用形容词或动词短语,如
Runnable,Comparable,Listable。 - 避免暴露实现细节:接口应抽象,不包含具体逻辑。
- 向后兼容:添加新方法时考虑默认方法(Java 8+),避免破坏现有实现。
- 文档注释:每个方法都要有清晰的 Javadoc。
- 慎用继承:优先组合而非继承接口。
- 标记接口 vs 功能接口:
- 标记接口(如
Serializable)无方法,仅做标识。 - 功能接口(如
Runnable)只有一个抽象方法,可用于 Lambda。
- 标记接口(如
💡 Java 8+ 推荐使用
@FunctionalInterface注解标注函数式接口。
5. HashMap 底层是 数组 + 链表 + 红黑树,为什么要用这几类结构?
🧩 结构设计目的:
| 结构 | 作用 | 为什么选它? |
|---|---|---|
| 数组 | 主容器,通过 hash 定位桶位置 | O(1) 时间复杂度访问 |
| 链表 | 解决哈希冲突(拉链法) | 简单、插入删除快 |
| 红黑树 | 当链表过长时转换为树,优化查找 | 保证最坏情况 O(log n),防止退化 |
🔄 转换条件(JDK 1.8+):
- 链表长度 ≥ 8 且 数组长度 ≥ 64 → 转为红黑树
- 红黑树节点 ≤ 6 → 退化为链表
🎯 为什么不用其他结构?
- 平衡二叉树:维护成本高,旋转频繁。
- 跳表:内存占用高,Java 未采用。
- 纯链表:极端情况下退化为 O(n)。
✅ 总结:HashMap 是“空间换时间 + 动态调整”的经典设计,兼顾平均性能与最坏情况。
6. 讲一讲 Java 对象的内存布局?
以 HotSpot VM 为例,对象在堆中分为三部分:
1️⃣ 对象头(Header)
- Mark Word:存储 hashCode、GC 分代年龄、锁标志位、线程 ID 等。
- Klass Pointer:指向方法区中类的元数据(类型信息)。
- (数组对象还有长度字段)
2️⃣ 实例数据(Instance Data)
- 存储对象真正的内容(字段值),按声明顺序排列(受对齐影响)。
3️⃣ 对齐填充(Padding)
- 为了满足 JVM 的内存对齐要求(通常是 8 字节倍数),补足空白。
📊 示例:一个空对象在 64 位 JVM 上通常占 16 字节(12 字节头 + 4 字节对齐)。
7. Java 对象为什么需要内存对齐?
✅ 原因:
-
CPU 访问效率:
- CPU 读取内存时,按“字长”(如 64 位机一次读 8 字节)进行。
- 如果数据未对齐,可能需要两次内存访问,降低性能。
-
硬件限制:
- 某些架构(如 ARM)不允许非对齐访问,会抛出异常。
-
原子性保障:
- 对齐后,对某个字段的读写可以是原子操作(尤其配合 volatile)。
🛠️ JVM 如何处理?
- 自动在对象末尾添加 padding,使总大小为 8 字节倍数。
- 字段重排序(在不影响语义前提下)以减少空洞。
💡 可通过
-XX:+PrintFieldLayout查看对象实际布局。
8. Java 对象内存分配是如何保证线程安全的?
主要依赖以下机制:
1️⃣ TLAB(Thread Local Allocation Buffer)
- 每个线程在 Eden 区预分配一块私有内存区域。
- 线程分配对象时先在 TLAB 中分配,无需加锁。
- TLAB 满了再竞争全局堆空间。
2️⃣ CAS + 失败重试
- 若 TLAB 不足,使用 CAS 操作尝试在全局堆上分配。
- 失败则自旋或进入同步队列。
3️⃣ 同步机制(极少用)
- 极端情况下,JVM 会使用互斥锁保证分配原子性。
✅ 总结:TLAB 是主流方案,极大提升了并发分配效率,几乎无锁。
9. 重载与覆盖的区别?
| 特性 | 重载(Overload) | 覆盖(Override) |
|---|---|---|
| 发生位置 | 同一个类内 | 子类重写父类方法 |
| 方法签名 | 参数列表不同(类型/数量/顺序) | 方法名、参数列表、返回类型必须相同 |
| 访问修饰符 | 无限制 | 不能比父类更严格(如 public → private) |
| 异常 | 无限制 | 不能抛出更多检查异常 |
| 绑定方式 | 编译时绑定(静态多态) | 运行时绑定(动态多态) |
| 关键字 | 无 | @Override(推荐) |
📌 示例:
// 重载
void print(String s) {}
void print(int i) {}
// 覆盖
class Parent { void show() {} }
class Child extends Parent { @Override void show() {} }
10. Java 语言中关键字 static 的作用是什么?
static 表示“属于类,不属于实例”。
✅ 主要用途:
1. 静态变量(类变量)
- 所有实例共享同一份拷贝。
- 生命周期随类加载而开始,卸载而结束。
- 常用于常量、计数器、配置等。
2. 静态方法(类方法)
- 可直接通过类名调用,无需创建对象。
- 不能访问非静态成员(因无 this 引用)。
- 常见工具方法:
Math.sqrt(),Collections.sort()
3. 静态代码块
- 类加载时执行一次,用于初始化静态资源。
static {
System.loadLibrary("nativeLib");
}
4. 静态内部类
- 不依赖外部类实例,可独立存在。
- 常用于构建器模式、辅助类。
⚠️ 注意事项:
- 静态方法不能被 override(只能 hide)。
- 静态上下文不能直接访问非静态成员。
- 过度使用 static 会导致代码难以测试和维护。
✅ 总结建议
这些题目覆盖了 Java 基础、集合、内存模型、面向对象、并发等核心知识点,是中大厂高频考点。建议你:
- 结合源码理解(如 HashMap、Object 头)
- 动手写小 demo 验证(如 static、重载覆盖)
- 记忆关键数字(如 HashMap 转树阈值 8、对象头大小 12/16 字节)
祝你面试顺利,拿到心仪 Offer!💼🚀
AtomGit 是由开放原子开源基金会联合 CSDN 等生态伙伴共同推出的新一代开源与人工智能协作平台。平台坚持“开放、中立、公益”的理念,把代码托管、模型共享、数据集托管、智能体开发体验和算力服务整合在一起,为开发者提供从开发、训练到部署的一站式体验。
更多推荐

所有评论(0)