大纲:
一、线程相关
① 进程与线程的区别:(重点掌握)
-----------------------------------------------------------------------------------------------
-----------------------------------------------------------------------------------------------
-----------------------------------------------------------------------------------------------
② 多线程与单线程的关系:(掌握)
③ 线程的状态有哪些?(掌握)
答:线程的状态包括 新建状态,运行状态,阻塞等待状态和消亡状态。其中阻塞等待状态又分为BLOCKED, WAITING和TIMED_WAITING状态。
所有的状态如下所示:
- NEW
- RUNNABLE
- BLOCKED
- WAITING
- TIMED_WAITING
- TERMINATED
线程的状态图
原文链接:https://blog.csdn.net/pange1991/article/details/53860651
④ 多线程编程中常用的函数比较
答:多线程编程中的常用函数的比较和特性总结如下。
sleep 和 wait 的区别:
- sleep方法:是Thread类的静态方法,当前线程将睡眠n毫秒,线程进入阻塞状态。当睡眠时间到了,会解除阻塞,进入可运行状态,等待CPU的到来。睡眠不释放锁(如果有的话)。
- wait方法:是Object的方法,必须与synchronized关键字一起使用,线程进入阻塞状态,当notify或者notifyall被调用后,会解除阻塞。但是,只有重新占用互斥锁之后才会进入可运行状态。睡眠时,会释放互斥锁。
join 方法:当前线程调用,则其它线程全部停止,等待当前线程执行完毕,接着执行。
yield 方法:该方法使得线程放弃当前分得的 CPU 时间。但是不使线程阻塞,即线程仍处于可执行状态,随时可能再次分得 CPU 时间。
解析:
这个题目主要是考察 sleep和wait方法所处的类是哪个,并且考察其在休眠的时候对于互斥锁的处理。
-------------------------------------------------------------------------------------------------------------------
二、线程池相关
① Java中的线程池有了解吗?
若不太理解,先查看链接:https://www.cnblogs.com/yuanfy008/p/10117878.html
线程池就是创建多个线程并且进行管理的容器。(线程池是个容器,可以创建线程和管理线程,并且给线程分配任务)
-------------------------------------------------------------------------------------------------------------------
-------------------------------------------------------------------------------------------------------------------
线程池参数的含义:
-------------------------------------------------------------------------------------------------------------------
线程池的种类:
1、newFixedThreadPool创建一个指定工作线程数量的线程池。每当提交一个任务就创建一个工作线程,如果工作线程数量达到线程池初始的最大数,则将提交的任务存入到任务队列中。
2、newCachedThreadPool创建一个可缓存的线程池。这种类型的线程池特点是:
- 1).工作线程的创建数量几乎没有限制(其实也有限制的,数目为Interger. MAX_VALUE), 这样可灵活的往线程池中添加线程。
- 2).如果长时间没有往线程池中提交任务,即如果工作线程空闲了指定的时间(默认为1分钟),则该工作线程将自动终止。终止后,如果你又提交了新的任务,则线程池重新创建一个工作线程。
3、newSingleThreadExecutor创建一个单线程化的Executor,即只创建唯一的工作线程来执行任务,如果这个线程异常结束,会有另一个取代它,保证顺序执行(我觉得这点是它的特色)。单工作线程最大的特点是可保证顺序地执行各个任务,并且在任意给定的时间不会有多个线程是活动的 。
4、newScheduleThreadPool创建一个定长的线程池,而且支持定时的以及周期性的任务执行,类似于Timer。
-------------------------------------------------------------------------------------------------------------------
线程池的排队策略:
当我们向线程池提交任务的时候,需要遵循一定的排队策略,具体策略如下:
- 如果当前运行的线程少于corePoolSize(核心线程数),则创建新线程来执行任务(注意,执行这一步骤需要获取全局锁) ;
- 如果运行的线程等于或多于corePoolSize,则将任务加入BlockingQueue(阻塞队列/任务队列);
- 如果无法将任务加入BlockingQueue(队列已满),则在非corePool中创建新的线程来处理任务(注意,执行这一步骤需要获取全局锁) ;
- 如果创建新线程将使当前运行的线程超出maximumPoolSize,任务将被拒绝,并执行线程饱和策略 ;
- 如果无法将任务加入BlockingQueue(队列已满),则在非corePool中创建新的线程来处理任务(注意,执行这一步骤需要获取全局锁) ;
参考链接:https://www.cnblogs.com/mengen/p/11890520.html
三、锁相关
① 原子性,可见性与有序性
参考链接:https://www.cnblogs.com/virgosnail/p/10470037.html
答:多线程环境下的线程安全主要体现在原子性,可见性与有序性方面。
② 谈谈你对synchronized关键字的理解
答: synchronized是Java中的一个关键字,是一个内部锁。它可以使用在方法上和方法块上,表示同步方法和同步代码块。在多线程环境下,同步方法或者同步代码块在同一时刻只允许有一个线程在执行,其余线程都在等待获取锁,也就是实现了整体并发中的局部串行。
内部锁底层实现:
- 进入时,执行monitorenter,将计数器+1,释放锁monitorexit时,计数器-1
- 当一个线程判断到计数器为0时,则当前锁空闲,可以占用;反之,当前线程进入等待状态
参考链接:https://blog.csdn.net/JinXYan/article/details/88836623
③ 谈谈你对volatile关键字的理解