- 每日小结
本栏目Java开发岗高频面试题主要出自以下各技术栈:Java基础知识
、集合容器
、并发编程
、JVM
、Spring全家桶
、MyBatis等ORMapping框架
、MySQL数据库
、Redis缓存
、RabbitMQ消息队列
、Linux操作技巧
等。
=======================================================================================
当线程A在运行一段代码的时,这时候另一个线程B也需要运行,但是在运行过程中的线程A执行完成之前,另一个线程B是无法获取到执行对象锁的,这个时候就会造成线程阻塞
。
sleep()
、wait()
调用后都会暂停当前线程并让出cpu的执行时间,但不同的是sleep不会释放当前持有的对象的锁资源,到时间后会继续执行
,而wait会放弃所有锁并需要notify/notifyAll后重新获取到对象锁资源后才能继续执行
。
java程序中出现线程阻塞的几种情况:
1、睡眠状态:
Thread.sleep (long millis)
方法,使线程转到阻塞状态。millis参数设定睡眠的时间,以毫秒为单位。当睡眠结束后,就转为就绪(Runnable)状态
。sleep()平台移植性好。
2、等待状态:
当一个线程正在运行时调用了wait()
方法,此时该线程需要交出CPU执行权,也就是将锁释放出去,交给另一个线程,该线程进入等待状态,但与睡眠状态不一样的是,进入等待状态的线程不需要设置睡眠时间
,但是需要执行notify()
或者notifyall()
来对其唤醒
,自己是不会主动醒来的,等被唤醒之后,该线程也会进入就绪状态,但是进入仅需状态的该线程手里是没有执行权的,也就是没有锁,而睡眠状态的线程一旦苏醒,进入就绪状态时是自己还拿着锁的。
3、礼让状态:
Thread.yield()
方法,暂停当前正在执行的线程对象,把执行机会让给相同或者更高优先级的线程
。yield() 使得线程放弃当前分得的 CPU 时间,但是不使线程阻塞,即线程仍处于可执行状态,随时可能再次分得 CPU 时间。调用 yield() 的效果等价于调度程序认为该线程已执行了足够的时间从而转到另一个线程。
4、自闭状态:
当一个线程正在运行时,调用了一个join()
方法,此时该线程会进入阻塞状态,另一个线程会运行,直到运行结束后,原线程才会进入就绪状态
。这个比较像是”走后门“,本来该先把你的事情解决完了再解决后边的人的事情,但是这时候有走后门的人,那就会停止给你解决,而优先把走后门的人事情解决了;
5、suspend() 和 resume() :
两个方法配套使用,suspend()使得线程进入阻塞状态,并且不会自动恢复,必须其对应的resume() 被调用,才能使得线程重新进入可执行状态。
典型地,suspend() 和 resume() 被用在等待另一个线程产生的结果的情形:测试发现结果还没有产生后,让线程阻塞,另一个线程产生了结果后,调用 resume() 使其恢复。Thread中suspend()和resume()两个方法在JDK1.5中已经废除
,因为有死锁倾向。
回答这个问题,我们得了解一下线程模型(下述对应关系为 线程
对内核调度实体
)
-
多对1用户级线程模型
-
1对1内核级线程模型
-
多对多两级线程模型
多对1用户级线程模型
-
线程的创建、调度、同步,由所属进程的用户空间线程库实现。
-
用户态线程,对内核几乎是透明的(许多操作不需要内核接管)
-
但线程总要有一些操作经过内核,比如系统调用。
-
不需要频繁的内核态/用户态切换,处理速度非常快。
-
该模式下,当进程的某个线程,系统调用(比如I/O)阻塞时,该进程也会阻塞。
该模式下,进程的所有线程,都对应一个内核调度实体(KES),多对一,并且内核不知道这个进程有哪些线程。KES无法将其他线程,调度到其他处理器上。该进程(所有的线程)被阻塞,直到本次系统调用(比如I/O)结束。
1对1内核级线程模型
-
每个用户线程都对应一个的内核调度实体。
-
内核会对每个线程进行调度,可以调度到其他处理器上。
-
线程每次操作会在用户态和内核态切换。
-
线程数量过多时,对系统性能有影响。
多对多两级线程模型
-
每个用户线程拥有多个内核调度实体
-
多个用户线程也可以对应一个内核调度实体
-
实现该模型非常复杂。
线程阻塞时,在多对1用户级线程模型下,会导致所属进程阻塞
。
在1对1或多对多模型下,不会导致进程阻塞
,目前linux基本上都采用一对一模型。
课间休息,又来秀一下来自咱们群里同学的搬砖工地,坐标:青岛。
作者:内蒙古上单
==============================================================================
阻塞和非阻塞关注的是程序在
等待调用结果
(消息,返回值)时的状态
。
阻塞调用
:指调用结果返回之前,当前线程会被挂起。调用线程只有在得到结果之后才会返回。
非阻塞调用
:指在不能立刻得到结果之前,该调用不会阻塞当前线程。
就像领导周末打电话问我能不能去公司加班,我支支吾吾半天领导没听懂我说的啥,如果是阻塞式调用,领导(线程)会把自己挂起
,一直追问我直到我告诉他能不能去。
最后
自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。
深知大多数Java工程师,想要提升技能,往往是自己摸索成长,自己不成体系的自学效果低效漫长且无助。
因此收集整理了一份《2024年Java开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Java开发知识点,不论你是刚入门Java开发的新手,还是希望在技术上不断提升的资深开发者,这些资料都将为你打开新的学习之门!
如果你觉得这些内容对你有帮助,需要这份全套学习资料的朋友可以戳我获取!!
由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!
识点,不论你是刚入门Java开发的新手,还是希望在技术上不断提升的资深开发者,这些资料都将为你打开新的学习之门!**
如果你觉得这些内容对你有帮助,需要这份全套学习资料的朋友可以戳我获取!!
由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!