并发篇:线程状态和线程池
线程状态
Java 线程六种状态
步骤(结合上图):
正常:
New新建——>start——>可运行状态——>操作系统交给cpu——>。。。。。——>完成运行,终结状态
阻塞:
cpu运行——>获取锁失败——>阻塞——>cpu运行阻塞代码——>当前面释放锁后,唤醒阻塞线程——>获取锁成功——>运行
等待:
cpu运行——>获取锁——>不满足某些条件——>等待状态,交给其它线程先运行——>满足条件后,另一个线程来唤醒等待线程——>运行
等待(有时限):
- 获取锁wait(long):(到期时间和notify两种唤醒)
long 毫秒值
-
cpu运行——>获取锁——>不满足某些条件——>等待状态,交给其它线程先运行——>时间到达,自己唤醒自己——>运行
-
cpu运行——>获取锁——>不满足某些条件——>等待状态,交给其它线程先运行——>满足条件后,另一个线程来唤醒等待线程——>运行
- sleep(long):
long 毫秒值
-
cpu运行——>睡眠——>等待状态,交给其它线程先运行——>时间到达,自己唤醒自己——>运行
正常执行线程
提示:main为主线程,t1为自定义的一个线程。
t1线程到达runnable 才可以执行,所以输出了 running… ,t1线程结束后,状态变为终结状态, 如果不理解的可以对着状态图来
阻塞
输出:
main先获取Lock, t2获取锁失败,处于阻塞,当主线main释放锁后, t2线程获取锁,输出in sync
等待
从操作层面上,Java线分为了五种状态
解释
- 分到CPU时间的:
- 运行可以分到CPU时间的:就绪
- 分不到CPU时间的:阻塞
- 运行态:分到 cpu 时间,能真正执行线程内代码的
- 就绪态:有资格分到 cpu 时间,但还未轮到它的
- 阻塞态:没资格分到 cpu 时间的
- 涵盖了 java 状态中提到的阻塞、等待、有时限等待
- 多出了阻塞 I/O,指线程在调用阻塞 I/O 时,实际活由 I/O 设备完成,此时线程无事可做,只能干等
- 新建与终结态:与 java 中同名状态类似,不再啰嗦
线程池
七大参数
- corePoolSize 核心线程数目 - 池中会保留的最多线程数
- maximumPoolSize 最大线程数目 - 核心线程+救急线程的最大数目
- keepAliveTime 生存时间 - 救急线程的生存时间,生存时间内没有新任务,此线程资源会释放
- unit 时间单位 - 救急线程的生存时间单位,如秒、毫秒等
- workQueue - 当没有空闲核心线程时,新来任务会加入到此队列排队,队列满会创建救急线程执行任务
- threadFactory 线程工厂 - 可以定制线程对象的创建,例如设置线程名字、是否是守护线程等
- handler 拒绝策略 - 当所有线程都在繁忙,workQueue 也放满时,会触发拒绝策略
- 抛异常 java.util.concurrent.ThreadPoolExecutor.AbortPolicy
- 由调用者执行任务 java.util.concurrent.ThreadPoolExecutor.CallerRunsPolicy
- 丢弃任务 java.util.concurrent.ThreadPoolExecutor.DiscardPolicy
- 丢弃最早排队任务 java.util.concurrent.ThreadPoolExecutor.DiscardOldestPolicy
执行步骤:
- 如果核心线程满了,加入队列中
- 核心线程执行完一个任务,从队列中取一个出来,队列就空了一个
- 如果队列满了,核心线程也满了,创建救急线程来执行,执行任务完后,此线程会在特定时间后释放线程资源
- 如果队列满了,核心线程,救急线程也满了,将采用拒绝策略:一共有四种
代码演示: