Java8面试深度解析:元空间与线程池实战

需积分: 5 0 下载量 143 浏览量 更新于2024-08-03 收藏 10KB MD 举报
"Java面试题集,包括Java8的变化以及如何自定义线程线程池" 在Java开发中,面试通常会涉及到各种技术点,包括语言特性、内存管理以及并发编程等方面。以下是对两个关键面试题的详细解答。 ### 1、Java8为什么要将永久代(PermGen)替换为元空间(MetaSpace)呢? **永久代**是Java虚拟机(JVM)早期版本中用于存储类元数据的空间,如类的常量池、字段信息和方法信息等。然而,由于以下几个原因,Java 8将其替换为**元空间**: 1. **内存限制**:永久代有一个固定的大小上限,由JVM参数`-XX:MaxPermSize`设置。这个上限可能导致`java.lang.OutOfMemoryError: PermGen space`异常,即使系统有更多可用内存也无法扩展。 2. **动态调整**:元空间使用的是直接内存,它的大小受本机可用内存限制,而非固定。这意味着元空间的大小可以随着应用的需要动态扩展,可以通过`-XX:MaxMetaspaceSize`来设置最大值,默认是无限制。 3. **避免内存溢出**:由于元空间不使用堆内存,因此不会因为类加载过多而导致堆内存溢出。 4. **减少垃圾收集开销**:元空间的垃圾收集相对于永久代更为高效,因为它只处理类加载器相关的对象,而不涉及其他对象的引用。 ### 2、如何自定义线程线程池? **线程池**是Java中管理并发执行的关键工具,通过`ThreadPoolExecutor`类实现。自定义线程池时,需要关注以下几个构造参数: - **corePoolSize**:核心线程数,即使没有任务,这些线程也会一直存活。 - **maximumPoolSize**:最大线程数,当工作队列满时,最多能创建的线程数量。 - **keepAliveTime**:非核心线程在空闲时的存活时间,达到这个时间后会被终止。 - **unit**:存活时间的时间单位。 - **workQueue**:阻塞队列,用于存放待执行的任务。 - **threadFactory**:线程工厂,用于创建新线程。 - **handler**:线程池拒绝策略,当工作队列已满且达到最大线程数时,如何处理新提交的任务。 下面是一个自定义线程池的代码示例: ```java ThreadPoolExecutor executor = new ThreadPoolExecutor( 1, // 核心线程数 2, // 最大线程数 60L, // 线程空闲60秒后被终止 TimeUnit.SECONDS, // 时间单位 new ArrayBlockingQueue<>(3), // 阻塞队列,容量为3 new ThreadFactory() { // 自定义线程工厂 @Override public Thread newThread(Runnable r) { return new Thread(r, "自定义线程"); } }, new ThreadPoolExecutor.AbortPolicy() // 拒绝策略,抛出异常 ); ``` 在这个例子中,我们创建了一个线程池,具有1个核心线程,最大2个线程,线程空闲60秒后被终止,工作队列是一个容量为3的ArrayBlockingQueue,线程工厂用于创建命名线程,拒绝策略是直接抛出异常。 了解这些核心概念,开发者可以根据实际需求调整线程池配置,以优化并发性能并避免资源浪费。在面试中,能够深入理解并解释这些细节将有助于展示你的专业技能。