Java 基础面试三

Java 基础面试三

Java 泛型

【中等】Java 泛型的作用是什么?

::: info Java 泛型是什么?

:::

泛型允许在类、接口、方法上使用类型参数(如 <T>,使代码能适应多种数据类型,同时保证类型安全。

::: info Java 泛型有什么用?

:::

  • 类型安全:编译时检查类型,避免运行时 ClassCastException
  • 代码复用:同一套逻辑可处理不同数据类型(如 List<String>List<Integer>)。
  • 消除强制转换:直接使用泛型类型,无需手动转换(如 (String) list.get(0))。

::: info Java 泛型有什么特性?

:::

  • 类型擦除:泛型仅在编译时有效,运行时类型信息会被擦除(List<String> 运行时变成 List)。
  • **通配符 <?>**:表示未知类型(如 List<?> 可接受任意类型的 List)。
  • 界限限定
    • T extends Class(限定类型范围,如 <T extends Number>)。
    • <? super T>(支持父类类型)。

简单示例

1
2
3
4
5
6
7
8
9
10
11
// 泛型类
class Box<T> {
private T content;
public void set(T content) { this.content = content; }
public T get() { return content; }
}

// 使用
Box<String> box = new Box<>();
box.set("Hello");
String value = box.get(); // 无需强制转换

一句话总结:泛型让代码更灵活、安全,减少冗余和运行时错误。

【中等】什么是 Java 泛型的上下界限定符?

Java 泛型的上下界限定符用于限制泛型类型参数的范围,确保类型安全,提供更灵活的类型约束。

::: info Java 什么是上界限定符?有什么用?

:::

上界限定符(<? extends T> 限定泛型类型必须是 T 或其子类T 可以是类或接口)。

特点

  • 只读安全:能安全读取数据(因为元素至少是 T 类型)。
  • 不能写入:无法确定具体子类型,防止类型污染。

示例

1
2
3
4
5
6
7
// 接受 Number 或其子类(如 Integer, Double)
void printList(List<? extends Number> list) {
for (Number num : list) { // 安全读取
System.out.println(num);
}
// list.add(1); // 编译错误!无法安全写入
}

::: info Java 什么是下界限定符?有什么用?

:::

下界限定符(<? super T>)限定泛型类型必须是 T 或其父类

特点

  • 可写入:能安全添加 T 及其子类的对象。
  • 读取受限:只能以 Object 类型读取(因为父类型不确定)。

示例

1
2
3
4
5
6
7
// 接受 Integer 或其父类(如 Number, Object)
void addNumbers(List<? super Integer> list) {
list.add(1); // 安全写入 Integer
list.add(2);
// Integer num = list.get(0); // 编译错误!需强制转换
Object obj = list.get(0); // 只能以 Object 读取
}

通配符限定对比

类型 语法 读取 写入 应用
上界 ? extends T 安全(作为T) 禁止 生产者场景
下界 ? super T 需转Object 安全(T及子类) 消费者场景
无界 ? 作为Object 禁止 完全不确定类型

小结

  • **extends T**:安全读取,限制类型上界。如遍历 List<? extends Number>
  • **super T**:安全写入,限制类型下界。如 Collections.copy(dest<? super T>, src<? extends T>)
  • PECS 原则(Producer-Extends, Consumer-Super)指导何时用哪种限定符。
    • 生产者(Producer)extends(输出数据)。
    • 消费者(Consumer)super(输入数据)。

【中等】泛型擦除的作用是什么?

泛型擦除是 Java 在编译时检查类型安全运行时丢弃类型信息的折中设计,平衡了兼容性、性能和类型安全,但牺牲了部分运行时灵活性。

泛型擦除是 Java 泛型的实现机制:

  • 编译时:泛型类型(如 <T>List<String>)会被检查,确保类型安全。
  • 运行时:所有泛型类型信息会被擦除,替换为原始类型(Raw Type)边界类型(如 Object/extends 上限)

泛型擦除规则

泛型定义 擦除后类型 示例
无界限 <T> Object List<T>List
有界限 <T extends Number> Number(边界类型) Box<T>Box<Number>
通配符 <?> / <? extends T> 边界类型 List<?>List
<? super T> Object List<? super Integer>List

泛型擦除作用

  • 兼容性:确保泛型代码能与旧版 Java(非泛型)字节码兼容。
  • 运行时效率:避免为每个泛型类型生成新类,减少 JVM 负担。
  • 简化设计:统一类型系统,避免 C++ 模板的复杂性。

泛型擦除的问题

  • 类型信息丢失:运行时无法获取泛型参数(如 List<String>List<Integer> 运行时都是 List)。

    1
    2
    List<String> list = new ArrayList<>();
    System.out.println(list.getClass()); // 输出 ArrayList,而非 ArrayList<String>
  • 强制类型转换:编译器自动插入类型转换代码。

    1
    2
    List<String> list = new ArrayList<>();
    String s = list.get(0); // 编译后实际为:(String) list.get(0)
  • 不支持原生类型:不能直接使用 List<int>,必须用包装类(如 List<Integer>)。

绕过擦除的限制

  • **显式传递 Class<T>**:通过反射保留类型信息。

    1
    2
    3
    <T> void create(Class<T> clazz) {
    T instance = clazz.newInstance(); // 运行时知道具体类型
    }
  • 类型令牌(Type Token):利用匿名子类捕获泛型类型。

    1
    new TypeToken<List<String>>() {};  // Guava 提供的方案

典型问题与解决方案

问题场景 解决方案
需要运行时获取泛型类型 传递 Class<T> 参数或使用 Type Token
泛型数组创建(new T[] 使用 Object[] 转换或反射(Array.newInstance
方法重载冲突(如 void foo(List<String>)void foo(List<Integer>) 编译报错(擦除后方法签名相同)

Java 反射

【简单】什么是反射?反射有什么作用?

反射(Reflection)是 Java 提供的动态机制,允许程序在运行时

  • 获取类的信息(类名、方法、字段、注解等)
  • 操作类的成员(调用方法、访问/修改字段、创建对象等)
  • 绕过访问控制(如调用私有方法)

反射核心类

  • Class<T>:表示类或接口
  • Method:表示类的方法
  • Field:表示类的字段
  • Constructor:表示类的构造方法

反射的主要用途

  • 动态加载类(如插件化开发)
  • 框架设计(如 Spring 的依赖注入、Hibernate 的 ORM 映射)
  • 测试工具(如 Mockito 模拟对象)
  • 绕过访问限制(调试或特殊场景)

如何使用反射?

::: code-tabs#反射使用示例

@tab 获取 Class 对象

1
2
3
4
5
6
7
8
9
// 方式1:通过类名.class
Class<String> strClass = String.class;

// 方式2:通过对象.getClass()
String s = "Hello";
Class<?> strClass2 = s.getClass();

// 方式3:通过Class.forName("全限定类名")
Class<?> strClass3 = Class.forName("java.lang.String"); // 需处理ClassNotFoundException

@tab 创建对象

1
2
3
4
5
6
7
// 方式1:直接调用无参构造(需强制类型转换)
Class<?> clazz = Class.forName("com.example.User");
User user = (User) clazz.newInstance(); // 已过时,推荐用 getConstructor()

// 方式2:调用带参构造
Constructor<?> constructor = clazz.getConstructor(String.class, int.class);
User user = (User) constructor.newInstance("Alice", 25);

@tab 调用方法

1
2
3
4
5
6
7
8
9
// 获取方法(需方法名 + 参数类型)
Method method = clazz.getMethod("setName", String.class);

// 调用方法(需对象实例 + 参数值)
method.invoke(user, "Bob"); // 相当于 user.setName("Bob")

// 调用静态方法
Method staticMethod = clazz.getMethod("staticMethod");
staticMethod.invoke(null); // 静态方法传 null

@tab 访问/修改字段

1
2
3
4
5
6
7
8
9
10
11
// 获取字段(包括私有字段)
Field field = clazz.getDeclaredField("name");

// 允许访问私有字段
field.setAccessible(true); // 关闭访问检查

// 读取字段值
String name = (String) field.get(user); // 相当于 user.name

// 修改字段值
field.set(user, "Charlie"); // 相当于 user.name = "Charlie"

@tab 获取注解信息

1
2
3
4
5
// 获取类/方法/字段上的注解
Annotation[] annotations = clazz.getAnnotations();
if (clazz.isAnnotationPresent(MyAnnotation.class)) {
MyAnnotation anno = clazz.getAnnotation(MyAnnotation.class);
}

:::

【简单】反射有什么优缺点?

优点 缺点
动态性高(运行时决定行为) 性能较差(比直接调用慢)
可访问私有成员(突破封装) 代码可读性降低
支持泛型擦除后的类型操作 安全隐患(如破坏单例)

性能优化建议

  • 缓存 Class/Method/Field 对象:避免重复反射调用。
  • **优先使用 getDeclaredXXX**:比 getXXX 更高效(不检查继承链)。
  • **限制 setAccessible(true)**:频繁调用影响性能。

注意事项

  • 反射可以破坏封装性(如修改 final 字段、调用私有方法)。
  • **慎用 setAccessible(true)**:可能导致安全漏洞(如绕过权限检查)。

::: tip 扩展

Java Reflection: Why is it so slow?

:::

【中等】什么是 Java 中的动态代理?

动态代理是一种在运行时动态创建代理对象的技术,允许在不修改原始类代码的情况下,增强或拦截目标对象的方法调用。

Java 动态代理通过 ProxyInvocationHandler 在运行时生成接口代理对象,非侵入式地实现方法拦截和功能增强,是 AOP 和框架设计的核心技术。

  • **java.lang.reflect.Proxy**:提供静态方法创建代理对象(核心方法:Proxy.newProxyInstance())。
  • **java.lang.reflect.InvocationHandler**:接口,实现代理逻辑(核心方法:invoke())。

动态代理的特点

  • 运行时生成:代理类在运行时动态生成,无需手动编写。
  • 基于接口:只能代理接口(不能代理普通类)。
  • 非侵入性:无需修改原始代码即可增强功能。

应用场景

  • AOP(面向切面编程):如日志、事务管理(Spring AOP 基于动态代理)。
  • 远程方法调用(RPC):如 Dubbo 的消费者代理。
  • 权限控制:拦截方法调用检查权限。

动态代理 vs 静态代理

对比项 动态代理 静态代理
生成时机 运行时动态生成 编译时手动编写
维护成本 低(自动适配接口) 高(需为每个类编写代理)
灵活性 高(通用逻辑集中处理) 低(逻辑分散)

局限性

  • 仅支持接口代理:不能代理普通类(CGLIB 可弥补此问题)。
  • 性能开销:反射调用比直接调用略慢(现代 JVM 已优化)。

扩展:CGLIB 动态代理

  • 原理:通过字节码技术生成目标类的子类代理。
  • 特点:可代理普通类,但无法代理 final 类/方法。

【中等】JDK 动态代理和 CGLIB 动态代理有什么区别?

JDK 动态代理 vs. CGLIB 动态代理:

代理类型 JDK 动态代理 CGLIB 代理
实现机制 基于接口,运行时生成代理类($Proxy0 基于继承,生成目标类的子类
技术依赖 Java 反射 API(Proxy类) ASM 字节码操作库
限制条件 目标类必须实现接口 无法代理 final 类/方法
可代理目标 只能代理接口 可代理普通类和接口

性能对比

维度 JDK 动态代理 CGLIB 代理
生成速度 较快(反射生成) 较慢(需操作字节码)
调用速度 反射调用,略慢 直接方法调用,更快
内存占用 较小 较大(生成子类)

:现代 JVM 对反射做了优化,JDK 代理性能差距已不明显。

使用示例

::: code-tabs#反射使用示例

@tab JDK 动态代理

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
// 要求:目标类必须实现接口
public interface UserService {
void save();
}

// 代理逻辑
InvocationHandler handler = (proxy, method, args) -> {
System.out.println("JDK 代理前置处理");
Object result = method.invoke(target, args);
System.out.println("JDK 代理后置处理");
return result;
};

UserService proxy = (UserService) Proxy.newProxyInstance(
target.getClass().getClassLoader(),
target.getClass().getInterfaces(), // 关键:需传入接口
handler
);

@tab CGLIB 代理

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
// 目标类无需实现接口
public class UserService {
public void save() { System.out.println("保存用户"); }
}

// 代理逻辑
Enhancer enhancer = new Enhancer();
enhancer.setSuperclass(UserService.class);
enhancer.setCallback((MethodInterceptor) (obj, method, args, proxy) -> {
System.out.println("CGLIB 代理前置处理");
Object result = proxy.invokeSuper(obj, args); // 直接调用父类方法
System.out.println("CGLIB 代理后置处理");
return result;
});

UserService proxy = (UserService) enhancer.create(); // 生成子类对象

:::

如何选择?

场景 推荐代理 理由
目标对象实现了接口 JDK 动态代理 轻量级,标准库支持
目标对象无接口 CGLIB 唯一选择
需要代理 final 方法 JDK 动态代理 CGLIB 无法代理 final 方法
高性能要求(如高频调用) CGLIB 直接方法调用更快
避免额外依赖 JDK 动态代理 CGLIB 需引入第三方库

主流框架的选择

  • Spring AOP
    • 默认使用 JDK 动态代理(如果目标有接口)
    • 无接口时自动切换为 CGLIB
    • 可通过 @EnableAspectJAutoProxy(proxyTargetClass=true) 强制使用 CGLIB
  • MyBatis:Mapper 接口代理使用 JDK 动态代理

一句话总结

  • JDK 动态代理:基于接口,反射实现,轻量但功能有限。
  • CGLIB:基于继承,字节码增强,功能强但有 final 限制。
  • 选择依据:目标是否有接口、性能需求、是否允许第三方依赖。

Java 注解

【中等】Java 中的注解原理是什么?

注解通过编译期处理(APT)或运行时反射实现元数据编程,其本质是特殊接口,由 JVM 或工具库按生命周期策略处理。

注解本质

  • 元数据标签:注解本质是继承自 java.lang.annotation.Annotation 的接口
  • 编译后保留策略:通过 @Retention 指定生命周期
    • SOURCE:仅保留在源码(如 @Override
    • CLASS:保留到字节码(默认)
    • RUNTIME:运行时可通过反射读取(如 @SpringBootApplication

核心处理机制

  • 编译期处理
    • APT(Annotation Processing Tool):在编译时生成代码(如 Lombok)
    • 编译器检查:如 @Override 验证方法重写
  • 运行时处理
    • 反射读取:通过 getAnnotation() 获取注解信息(如 Spring 扫描 @Component
    • 动态代理:结合 AOP 实现功能增强(如 @Transactional

关键技术点

  • 元注解:修饰注解的注解(如 @Target 指定作用目标)
  • 注解属性:本质是接口方法(需编译时常量值)
  • 字节码操作:ASM 等工具可直接修改字节码中的注解信息

应用场景

  • 框架配置:Spring 的 @Autowired@RequestMapping
  • 代码生成:Lombok 的 @Data
  • 静态检查@Nullable@Deprecated

Java SPI

【中等】什么是 SPI,有什么用?

SPI 通过接口+配置文件实现运行时服务发现,是解耦和扩展的利器,JDBC/日志等经典框架均基于此机制。

SPI 是 Java 提供的服务发现机制,通过接口与实现分离,实现:

  • 运行时动态加载实现类
  • 解耦接口与实现
  • 可插拔式扩展

核心组成

组件 作用 示例
接口 定义服务标准 java.sql.Driver
实现类 提供具体功能 com.mysql.cj.jdbc.Driver
配置文件 声明实现类 META-INF/services/接口全限定名

工作原理

  • META-INF/services/下创建以接口全限定名命名的文件
  • 文件中写入实现类全限定名(每行一个)
  • 通过ServiceLoader动态加载实现类

主要应用场景

  • JDBC 驱动加载DriverManager
  • 日志门面实现(SLF4J → Logback/Log4j)
  • Spring Boot 自动配置
  • Dubbo 扩展点机制

优势与局限

优势 局限
实现热插拔 配置文件需严格规范
解耦接口与实现 原生SPI会加载所有实现类(可能浪费资源)
扩展性强 无默认实现筛选机制

与API的区别

维度 SPI API
调用方向 由实现方提供,调用方选择 由提供方定义,调用方使用
控制权 调用方控制 提供方控制
典型场景 JDBC驱动、日志实现 Java标准库

改进方案

  • Dubbo SPI:增加按需加载、扩展点缓存等优化
  • Spring FactoriesMETA-INF/spring.factories机制

Java IO

【简单】什么是序列化?什么是反序列化?

基本概念

  • 序列化:将对象转换为字节流(用于存储/传输)
  • 反序列化:将字节流恢复为对象

核心用途

  • 持久化存储(如保存到文件/数据库)
  • 网络传输(如RPC调用)
  • 深拷贝实现(通过序列化+反序列化)

Java实现方式

方式 特点 示例
Serializable接口 标记接口,默认Java序列化 class User implements Serializable
Externalizable接口 需手动实现读写逻辑 覆盖writeExternal()/readExternal()
第三方库(JSON/Protobuf等) 跨语言、高效 Gson、Jackson、Protobuf

关键注意事项

  • **serialVersionUID**:显式声明版本号,避免反序列化失败

    1
    private static final long serialVersionUID = 1L;
  • 敏感字段处理:用transient跳过序列化

    1
    private transient String password;  // 不会被序列化
  • 性能优化

    • 避免序列化大对象
    • 第三方库(如Protobuf)比Java原生序列化更快

常见序列化协议对比

协议 语言支持 可读性 性能 典型应用
Java原生 仅Java Java RMI
JSON 多语言 Web API
Protobuf 多语言 gRPC
Hessian 多语言 Dubbo

安全风险

  • 反序列化漏洞:恶意字节流可触发代码执行(需校验数据来源)
  • 解决方案
    • 使用白名单控制反序列化类
    • 替换为JSON等文本协议

【中等】Java 提供了哪些 IO 方式?

Java 提供了多种 I/O(输入输出)方式,主要分为 传统 I/O(BIO)、NIO(New I/O)、AIO(异步 I/O) 三大类,并支持 文件操作、网络通信、序列化 等场景。以下是主要 I/O 方式的概述及要点:

::: info 什么是 BIO?

:::

传统 I/O(BIO,Blocking I/O)是同步阻塞式 I/O,适用于连接数较少、延迟不敏感的场景。

核心类

  • 字节流InputStream / OutputStream(如 FileInputStreamFileOutputStream
  • 字符流Reader / Writer(如 FileReaderFileWriter
  • 缓冲流BufferedReaderBufferedWriter(提升性能)
  • 标准 I/OSystem.in(输入)、System.out(输出)

示例

1
2
3
4
5
6
try (BufferedReader reader = new BufferedReader(new FileReader("file.txt"))) {
String line;
while ((line = reader.readLine()) != null) {
System.out.println(line);
}
}

缺点:每个连接需要独立的线程,高并发时资源消耗大。

::: info 什么是 NIO?

:::

NIO(New I/O,Non-blocking I/O)是同步非阻塞 I/O,基于 通道(Channel)缓冲区(Buffer),支持多路复用(Selector)。

核心类

  • BufferByteBufferCharBuffer(数据存储)
  • ChannelFileChannelSocketChannelServerSocketChannel(数据传输)
  • Selector:监听多个通道的事件(如连接、读、写)

示例(NIO 文件复制)

1
2
3
4
try (FileChannel src = FileChannel.open(Paths.get("src.txt"));
FileChannel dest = FileChannel.open(Paths.get("dest.txt"), StandardOpenOption.CREATE, StandardOpenOption.WRITE)) {
src.transferTo(0, src.size(), dest);
}
  • 优点:单线程可处理多个连接,适合高并发(如 Netty 框架底层)。
  • 缺点:编程复杂度较高。

::: info 什么是 AIO?

:::

AIO(Asynchronous I/O)是异步非阻塞 I/O,基于回调或 Future 机制,适用于高吞吐场景。

核心类

  • AsynchronousFileChannel(文件操作)
  • AsynchronousSocketChannel(网络通信)
  • CompletionHandler(回调接口)

示例(AIO 文件读取)

1
2
3
4
5
6
7
8
9
10
11
12
AsynchronousFileChannel fileChannel = AsynchronousFileChannel.open(Paths.get("file.txt"));
ByteBuffer buffer = ByteBuffer.allocate(1024);
fileChannel.read(buffer, 0, buffer, new CompletionHandler<Integer, ByteBuffer>() {
@Override
public void completed(Integer result, ByteBuffer attachment) {
System.out.println("Read " + result + " bytes");
}
@Override
public void failed(Throwable exc, ByteBuffer attachment) {
exc.printStackTrace();
}
});
  • 优点:真正异步,适合长连接、高吞吐场景(如大文件传输)。
  • 缺点:JDK 实现较少,Linux 支持有限(底层依赖 epoll)。

::: info 有哪些常见的 IO 工具?

:::

  • 序列化ObjectInputStream / ObjectOutputStream(Java 原生序列化)
  • 压缩流GZIPInputStreamZipOutputStream
  • 内存映射文件MappedByteBuffer(NIO 高性能文件访问)
  • Files 工具类(Java 7+):
    1
    Files.readAllLines(Paths.get("file.txt")); // 快速读取文件

::: info BIO vs. NIO vs. AIO?

:::

类型 模型 适用场景 典型框架
BIO 同步阻塞 低并发、简单 I/O Java Socket
NIO 同步非阻塞 高并发、网络通信 Netty、Tomcat NIO
AIO 异步非阻塞 高吞吐、大文件操作 较少使用

选择建议

  • BIO:简单文件操作或低并发场景。
  • NIO:高并发网络编程(如 Netty)。
  • AIO:需要真正异步 I/O 的场景(但实际使用较少)。

如果需要更高层次的封装,可以考虑 Apache Commons IOGuava 等工具库。

【困难】NIO 如何实现多路复用?

::: info Java NIO 的核心组件有哪些?

:::

Java NIO 多路复用的核心是通过 Selector 轮询事件 + 非阻塞 Channel + Buffer 数据交换,允许单线程管理多个通道的 I/O 操作。这是构建高性能网络应用的基础,也是 Netty 等框架的底层原理。

Java NIO 核心组件

  • Selector(选择器):核心多路复用器,可监控多个 Channel 的 I/O 事件(如连接、读、写)
    • 通过 Selector.open() 创建
    • 一个 Selector 可绑定多个 Channel
  • Channel(通道):非阻塞 I/O 操作的抽象,支持读写。主要类型:
    • SocketChannel:TCP 网络通信
    • ServerSocketChannel:监听 TCP 连接
    • FileChannel:文件 I/O(不支持 Selector)
  • Buffer(缓冲区):数据容器(如 ByteBuffer),Channel 通过 Buffer 读写数据。

::: info Java NIO 多路复用的实现步骤是怎样的?

:::

多路复用实现步骤

(1) 创建 Selector 并注册 Channel

1
2
3
4
Selector selector = Selector.open();
ServerSocketChannel serverChannel = ServerSocketChannel.open();
serverChannel.configureBlocking(false); // 必须设为非阻塞
serverChannel.register(selector, SelectionKey.OP_ACCEPT); // 注册监听事件

(2) 事件类型

  • SelectionKey.OP_ACCEPT:接受连接(ServerSocketChannel
  • SelectionKey.OP_CONNECT:连接就绪(SocketChannel
  • SelectionKey.OP_READ:数据可读
  • SelectionKey.OP_WRITE:数据可写

(3) 事件轮询

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
while (true) {
int readyChannels = selector.select(); // 阻塞直到有事件就绪
if (readyChannels == 0) continue;

Set<SelectionKey> selectedKeys = selector.selectedKeys();
Iterator<SelectionKey> keyIterator = selectedKeys.iterator();

while (keyIterator.hasNext()) {
SelectionKey key = keyIterator.next();

if (key.isAcceptable()) {
// 处理新连接
} else if (key.isReadable()) {
// 处理读事件
} else if (key.isWritable()) {
// 处理写事件
}

keyIterator.remove(); // 必须移除已处理的键
}
}

::: info Java NIO 的关键机制有哪些?

:::

(1) 非阻塞模式

  • Channel 必须设置为非阻塞:channel.configureBlocking(false)
  • 避免单线程因 I/O 操作阻塞

(2) 事件驱动

  • Selector 通过操作系统级轮询(如 Linux 的 epoll)监听事件
  • 仅处理活跃的 Channel,避免无效遍历

(3) SelectionKey

  • 绑定 Channel 与 Selector 的关系
  • 可通过 key.attachment() 附加自定义对象(如会话状态)

::: info Java NIO 的底层原理是什么?

:::

  • Linux:基于 epoll 实现(高效监控大量文件描述符)
  • Windows:基于 IOCP(完成端口)
  • 相比传统 BIO 的线程池模型,NIO 单线程可处理数千连接

NIO 优点

  • 单线程管理多连接,资源消耗低
  • 高并发支持(如 Netty 框架底层依赖 NIO)
  • 避免线程上下文切换开销

NIO 适用场景

  • 高并发网络服务(如聊天服务器、API 网关)
  • 需要长连接的应用(如 WebSocket)
  • 大数据量、低延迟的 I/O 操作

Java 语法糖

【中等】Java 中有哪些常见的语法糖?

语法糖(Syntactic sugar) 代指的是编程语言为了方便程序员开发程序而设计的一种特殊语法,这种语法对编程语言的功能并没有影响。实现相同的功能,基于语法糖写出来的代码往往更简单简洁且更易阅读。

Java 中最常用的语法糖主要有泛型、自动拆装箱、变长参数、枚举、内部类、增强 for 循环、try-with-resources 语法、lambda 表达式等。所有这些语法糖在编译阶段都会被”脱糖”(desugar),即转换为更基础的Java语法结构。可以使用javap -c命令查看字节码来验证这一点。语法糖虽然不增加语言功能,但能显著提高代码的可读性和编写效率,是Java语言不断演进的重要组成部分。

自动装箱与拆箱 (Autoboxing/Unboxing)

1
2
3
4
5
// 自动装箱
Integer i = 10; // 实际编译为 Integer.valueOf(10)

// 自动拆箱
int n = i; // 实际编译为 i.intValue()

增强 for 循环 (foreach)

1
2
3
4
5
6
7
8
9
10
List<String> list = Arrays.asList("a", "b", "c");
// 语法糖形式
for (String s : list) {
System.out.println(s);
}
// 实际编译为迭代器模式
for (Iterator<String> it = list.iterator(); it.hasNext();) {
String s = it.next();
System.out.println(s);
}

变长参数 (Varargs)

1
2
3
4
5
6
7
public void print(String... args) {
for (String arg : args) {
System.out.println(arg);
}
}
// 实际编译为数组参数
public void print(String[] args) { ... }

数值字面量下划线

1
int million = 1_000_000;  // 编译后等同于 1000000

字符串拼接

1
2
3
4
5
6
7
8
9
String s = "a" + "b" + "c";
// 编译优化为
String s = "abc";

// 变量拼接会转为 StringBuilder
String a = "a", b = "b";
String result = a + b;
// 编译为
String result = new StringBuilder().append(a).append(b).toString();

switch 支持字符串 (Java 7+)

1
2
3
4
5
6
7
String fruit = "apple";
switch (fruit) {
case "apple":
System.out.println("It's an apple");
break;
// 实际编译为基于hashCode()和equals()的比较
}

默认构造方法

1
2
public class Person {}
// 如果没有显式定义构造方法,编译器会自动添加无参构造方法

枚举类 (Java 5+)

1
2
enum Color { RED, GREEN, BLUE }
// 实际编译为继承java.lang.Enum的类

内部类访问外部类成员

1
2
3
4
5
6
7
8
class Outer {
private int x = 10;
class Inner {
void print() {
System.out.println(x); // 实际通过 Outer.this.x 访问
}
}
}

方法引用 (Java 8+)

1
2
3
4
List<String> list = Arrays.asList("a", "b", "c");
list.forEach(System.out::println);
// 编译为lambda表达式
list.forEach(s -> System.out.println(s));

钻石操作符 (Diamond Operator, Java 7+)

1
2
3
List<String> list = new ArrayList<>();  // 类型推断
// Java 7之前需要
List<String> list = new ArrayList<String>();

集合字面量 (Java 9+ 的List.of等)

1
2
3
List<String> list = List.of("a", "b", "c");
Set<Integer> set = Set.of(1, 2, 3);
Map<String, Integer> map = Map.of("a", 1, "b", 2);

Lambda 表达式 (Java 8+)

1
2
3
// Lambda表达式
Runnable r = () -> System.out.println("Hello");
// 实际生成实现Runnable的匿名类

try-with-resources (Java 7+)

1
2
3
4
try (InputStream is = new FileInputStream("file.txt")) {
// 使用资源
} // 自动调用close()
// 编译为try-finally块

接口中的默认方法和静态方法 (Java 8+)

1
2
3
4
5
6
7
8
9
interface MyInterface {
default void defaultMethod() {
System.out.println("Default method");
}

static void staticMethod() {
System.out.println("Static method");
}
}

记录类 (Record, Java 14+)

1
2
3
4
5
6
record Point(int x, int y) {}
// 编译后自动生成:
// - 私有final字段x和y
// - 公共构造方法
// - 访问器方法x()和y()
// - equals(), hashCode(), toString()

instanceof 模式匹配

1
2
3
4
if (obj instanceof String s) {
// 可以直接使用s
System.out.println(s.length());
}

文本块 (Text Blocks, Java 15+)

1
2
3
4
5
6
7
String html = """
<html>
<body>
<p>Hello, world</p>
</body>
</html>
""";