Java线程池高频面试题总结
作者:灰小猿 发布时间:2023-10-22 02:12:34
1、在启动线程时,为什么要通过调用方法start执行方法run,而不能直接执行方法run?
调用方法start执行方法run,才是多线程的工作,**如果直接执行方法run,会被当成一个主线程下的普通方法执行,**而不会在某个线程中执行,因此不是多线程工作。
2、方法sleep、join和yield的区别有哪些?
方法sleep的作用是使当前线程暂停执行一段时间,让其他线程有机会继续执行;
方法join的作用是阻塞调用该方法的线程,直到当前线程执行完毕后,调用该方法的线程才能继续执行;
方法yield的作用是暂停当前正在执行的线程对象,并执行其他线程。
3.为什么方法wait、notify和notifyAll在Object类定义,而不在Thread类定义?
其中主要的原因是:
方法 wait、notify 和 notifyAll 不只是普通方法或同步工具,它们还是 Java 中两个线程之间的通信机制。对语言设计者而言, 如果不能通过 Java 关键字(例如 synchronized)实现通信机制,同时又要确保这个机制对每个对象可用,那么 Object 类则是的合理的声明位置。每个对象都可上锁,因此方法 wait 和 notify 在 Object 类而不是 Thread 类定义。
4、终止线程应该使用什么方法?为什么不推荐使用stop和destroy方法来终止线程?
终止线程应该使用方法interrupt,使用方法stop会带来两个问题,
一是清理性工作无法完成;
二是会立即释放所有锁,导致对象状态不一致。因此会造成不安全。
而方法destroy除了抛出NoSuchMethodError 以外没有做任何事情,因此无法终止线程。
5、什么是线程池?
线程池是一种线程的使用模式,创建若干个可执行的线程放入一个池(容器)中,有任务需要处理时,会提交到线程池中的任务队列,处理完后线程并不会被销毁,而是仍然在线程池中等待下一个任务。
6、追问:线程池的好处有哪些?
使用线程池有以下三点好处:
降低资源消耗,重复利用线程池中已经创建的资源,可以避免频繁的创建和销毁线程,从而减少资源的消耗。
提高响应速度,由于线程池中有已经创建的线程,因此当任务到达时可以直接执行。不需要等待线程的创建。
提高线程的可管理性,线程是稀缺资源,如果无限制的创建,不仅会消耗系统资源,还会降低系统的稳定性,使用线程池可以进行统一分配、调优和监控。
7、说一说线程池是如何创建的?
线程池的创建可以通过ThreadPoolExecutor类,
ThreadPoolExecutor类有4个构造方法,其中最一般话的构造方法包含7个参数。
public ThreadPoolExecutor(int corePoolSize, int maximumPoolSize, long keepAliveTime, TimeUnit unit, BlockingQueue<Runnable> workQueue, ThreadFactory threadFactory, RejectedExecutionHandler handler)
这7个参数的含义分别是:
corePoolSize
: 核心线程数,定义了最少可以同时运行的线程数量,当有新的任务时就会创建一个线程执行任务,当线程池中的线程数量达到corePoolSize 之后,到达的任务进入阻塞队列maximumPoolSize
: 最大线程数,定义了线程池中最多能创建的线程数量。keepAliveTime
: 等待时间,当线程池中的线程数量大于 corePoolSize 时,如果一个线程的空闲时间达到keepAliveTime 时则会终止,直到线程池中的线程数不超过 corePoolSize。unit
: 参数keepAliveTime 的单位。workQueue
: 阻塞队列,用来存储等待执行的任务。threadFactory
: 创建线程的工厂。handler
: 当拒绝处理任务时的策略。
8、向线程池提交任务的流程是什么?
如果正在运行的线程数量小于corePoolSize,则创建核心线程运行这个任务。
如果正在运行的线程数量大于或等于corePoolSize,则将这个任务放入阻塞队列。
如果阻塞队列满了,而且正在运行的线程数量小于maximumPoolSize,则创建非核心线程运行这个任务
如果阻塞队列满了,而且正在运行的线程数量大于或等于 maximumPoolSize,则线程池抛出RejectExecutionException 异常。
9、追问:说一说核心线程和非核心线程的区别?
核心线程和非核心线程的最大数目在创建线程时被锁定,核心线程和非核心线程的区别如下:
向线程池提交任务时,首先创建核心线程运行任务,直到核心线程数达到上限,然后将任务放入阻塞队列。
只有在核心线程数到达上限,且阻塞队列满的情况下,才会创建非核心线程运行任务。
10、如何关闭线程池?
可以通过调用线程池的方法shutdown或shutdownNow关闭线程池,
这两个方法的原理是 遍历线程池中的工作线程,对每个工作线程调用 interrupt 方法中断线程,无法响应中断的任务可能永远无法终止。
方法 shutDown 和 shutDownNow 有以下区别。
方法 shutDown 将线程池的状态设置成 SHUTDOWN,正在执行的任务继续执行,没有执行的任务将中断。
方法 shutDownNow 将线程池的状态设置成 STOP,正在执行的任务被停止,没有执行的任务被返回。
来源:https://blog.csdn.net/weixin_44985880/article/details/119563211
猜你喜欢
- 在8 里面Lambda是最火的主题,不仅仅是因为语法的改变,更重要的是带来了函数式编程的思想,我觉得优秀的程序员,有必要学习一下函数式编程的
- 用途项目中使用了 dubbo,注册中心使用的 zookeeper,使用 zookeeper 实现了一个简单的分布式锁(依赖 curator)
- 1. 定义在不破坏封装性的前提下,捕获一个对象的内部状态,并在该对象之外保存这个状态。这样以后就可将该对象恢复到原先保存的状态。2. 使用的
- 前言Spark Sql可以通过UDF来对DataFrame的Column进行自定义操作。在特定场景下定义UDF可能需要用到Spark Con
- 本文实例为大家分享了java web实现简单留言板的具体代码,供大家参考,具体内容如下一、目标用户可以登录并记住密码进入留言板,添加留言,点
- 我已经很精简了,两篇(Spring Boot启动过程(一)、spring Boot启动过程(二))依然没写完,接着来。refreshCont
- 需求背景最近的一个项目,在项目基本完工的阶段,客户提出要将所有业务操作的日志记录到数据库中,并且要提取一些业务的关键信息(比如交易单号)体现
- 开发中有时候需要自己封装分页排序时,List如何对某一属性排序呢,分享一个小实例,大家共勉,希望能对大家有用,请多多指教。1.Student
- 在很多场景下,maven不能直接访问到外网时,使用代理是其中常见的一种方式。这篇文章整理一下常见的maven中设置代理的方法。代理服务器代理
- 概念尽量使用合成/聚合,而不是使用继承实现复用。所谓的合成/聚合是指一个对象里持有另外一个类的对象,通过调用这些对象的方法得到复用已有功能的
- 前言基本语法首先我们要知道java的基础语法。1.由26个英文字母大小写,0-9,_或$组成2.数字不可以开头3.不可以使用关键字和保留字,
- 个人理解:把一个类里的多个命令分离出来,每个类里放一个命令,实现解耦合,一个类只对应一个功能,在使用命令时由另一个类来统一管理所有命令。缺点
- 本文实例讲述了Java Socket实现传输压缩对象的方法。分享给大家供大家参考,具体如下:前面文章《Java Socket实现的传输对象功
- 此处网上最多的做法是需要修改tomcat的参数配置大致如下:<Connector port="8080" prot
- 前言现在的项目一般是拆分成一个个独立的模块,当在其他项目中想要使用独立出来的这些模块,只需要在其pom.xml使用<dependenc
- 1 关于自动内存管理Java是由jvm来管理内存,包括自动分配以及自动回收,因此它不容易出现内存泄漏和内存溢出问题。C/C++,由程序员手动
- 前言RSA加密算法是一种非对称加密算法,简单来说,就是加密时使用一个钥匙,解密时使用另一个钥匙。因为加密的钥匙是公开的,所又称公钥,解密的钥
- 字节流和字符流对于文件必然有读和写的操作,读和写就对应了输入和输出流,流又分成字节和字符流。1.从对文件的操作来讲,有读和写的操作——也就是
- package com.wanmei.meishu;import java.io.FileInputStream;import java.i
- 前言痛点:在java开发的过程中,我们经常要面对各种各样的环境,比如开发环境,测试环境,正式环境,而这些环境对项目的需求也不相同。在此之前,