面试突击1

1.当线程没有拿到资源时,用户态和内核态的一个切换

操作系统中,进程线程是执行程序的基本单位。为了管理这些单位,操作系统使用了一种称为“进程状态”的机制,其中包括用户态和内核态两种状态。这两种状态代表了进程或线程在执行时的不同权限和上下文。

用户态(User Mode): 当进程或线程在用户态下运行时,它们只能访问受限的资源,例如它们自己的内存空间、文件、网络等。这是为了保护系统的稳定性和安全性,防止进程或线程意外地或恶意地访问其他进程或系统的核心资源。
**内核态(Kernel Mode):**当进程或线程需要执行一些特权操作,如访问硬件、修改其他进程的内存或执行系统调用时,它们会切换到内核态。在内核态下,进程或线程具有更高的权限,可以访问系统的核心资源。
现在,关于线程没有拿到资源时从内核态到用户态的转换:

当线程尝试获取某个资源(如内存、文件、网络套接字等)但未能成功时(例如,由于资源不足或资源已被其他线程占用),线程可能会执行一个系统调用,请求操作系统帮助获取该资源。这个系统调用会导致线程从用户态切换到内核态。在内核态下,操作系统会检查资源的情况,并决定是否满足线程的请求。如果资源不可用或无法满足线程的需求,操作系统会返回一个错误码给线程,告诉它无法获取资源。

一旦线程收到了这个错误码,它就知道它没有成功获取资源。然后,操作系统会将线程从内核态切换回用户态,让线程继续执行。

这种从用户态到内核态的转换和切换是操作系统管理和调度进程或线程的基本机制之一。它允许操作系统对进程或线程的行为进行监控和控制,确保系统的稳定性和安全性。

2.Synchronized

1.synchronized能保证原子性,有序性,可见性:

2.1加锁原理:

synchronized加锁原理:使用synchronized之后,当执行同步代码块前首先要先执行monitorenter指令,退出的时候monitorexit指令 ,其关键就是必须要对对象的监视器monitor进行获取 ,当线程获取monitor后才能继续往下执行,否则就只能等待。而这个获取的过程是互斥的,即同一时刻只有一个线程能够获取到monitor 。

2.2monitor 到底是什么?

Monitor在Java中是一个用于实现线程同步的机制,通常与synchronized关键字关联。Monitor内部包含了一个对象头(Object Header)和一块同步代码块。当线程尝试进入同步代码块时,它首先需要获得对象的monitor锁。如果锁被其他线程持有,则当前线程会被阻塞,直到获得锁为止。

具体来说,1.当一个线程尝试进入synchronized代码块时,它会尝试获取对象的锁。这个锁的状态信息存储在对象的对象头中。如果锁是偏向锁或轻量级锁,并且可以由当前线程获取,那么线程就会继续执行同步代码块。——>2.``如果锁已经升级为重量级锁,或者当前线程无法获取轻量级锁,那么就会涉及到Monitor的机制

在这种情况下,对象的对象头会被修改为指向一个Monitor对象(在JVM内部实现),这个Monitor对象包含了等待队列和锁的所有者信息。当前线程如果无法获取锁,就会被放入Monitor的等待队列中阻塞等待。当锁被释放时,Monitor会负责唤醒等待队列中的一个线程,并让它尝试获取锁。

2.2.1对象头里面有什么?

1.Mark Word:用于存储对象自身的运行时数据,如·哈希码(HashCode)、GC分代年龄、锁状态标志线程持有的锁偏向线程ID偏向时间戳等。这些信息都是用于Java对象的内存管理和并发控制。
2.Klass Word (或称为元数据类型指针):指向对象的类元数据(也就是方法区中的类型信息),虚拟机通过这个指针来确定这个对象是哪个类的实例。
可以通过这个指针——>对应的
类对象
(这个类对象可以找到InstanceKlass)
——>字节码文件加载到内存时就会得到Class对象(也就是镜像类)
,里面有InstanceKlass的地址,指向InstanceKlass(保存在方法区),从而获取字节码文件中的内容(也就是各种类信息)

2.2.2KlassWord指针和反射的区别:

1.反射:是Java提供的一种能力,允许程序在运行时获取和操作对象的内部属性、方法等信息。它通常涉及到使用java.lang.reflect包中的类,如Class、Method、Field等;它允许程序在运行时动态地访问和操作这些类元数据。例如,通过反射,您可以获取一个对象的Class对象然后进一步获取该类的所有方法、字段、构造函数等信息,甚至可以动态地调用方法和访问字段。

2.KlassWord:过程是Java对象实例化时的一部分,涉及到对象头的Klass Word(或称为元数据类型指针)和类元数据的加载。这个过程发生在对象创建时,由JVM自动处理,用于确定对象所属的类,以及该类在方法区中的元数据。这是Java对象模型的一部分,用于支持对象的类型识别和类型安全。
在这里插入图片描述

2.3 synchronized并发获取锁的过程

在这里插入图片描述

2.3.1 在多个线程竞争synchronized锁资源时,是公平还是非公平?那你是如何理解这种现象

synchronized关键字的非公平性确实可能导致线程饥饿的情况。当一个线程尝试获取synchronized锁时,它不会等待其他已经等待很久的线程,而是会立即尝试获取锁。如果它成功获取了锁,那么即使有其他线程已经在等待队列中等待了很长时间,它们也必须继续等待

2.4 synchronized的优化

2.4.1首先阐述 偏向锁和轻量型锁以及重量型锁之间的转换:

1.偏向锁(Biased Locking):
当同步代码块第一次被访问时,JVM会尝试使用偏向锁。在对象头中记录当前线程的ID,作为偏向锁。偏向锁是为了减少无竞争的锁获取和释放的开销。如果同步代码块在后续的执行中没有被其他线程访问,那么偏向锁就会持续保持。

2.轻量级锁(Lightweight Locking):
同步代码块出现竞争时(即多个线程尝试同时访问),偏向锁会升级为轻量级锁(轻量级锁是通过在对象头中存储一个指向线程栈锁记录指针来实现的——>如果此时原来的线程不再持有该锁新的线程通过CAS操作尝试将对象头的锁记录指针指向自己的锁记录,从而获取锁)。轻量级锁是为了减少线程挂起和恢复的开销。在轻量级锁下,线程会尝试通过自旋(忙等待)的方式获取锁,而不是直接阻塞。如果自旋成功,则线程获得锁并执行同步代码块。如果自旋失败(即锁仍被其他线程持有),则轻量级锁可能会进一步升级为重量级锁(锁膨胀)。

在这里插入图片描述
Cas的本质就是比较对象头并且交换(有性能损耗,看对象头中的锁状态是00否);——>就像反书包一样,每次都得翻一下书包看一下是否是自己名字

3.重量级锁(Heavyweight Locking):
当轻量级锁自旋超过一定的次数(通常取决于JVM的实现和配置)或者有其他线程在等待获取锁时,轻量级锁会升级为重量级锁。重量级锁会导致线程阻塞,并在需要时由操作系统进行调度。重量级锁的开销相对较大,因为它涉及到线程的挂起和恢复。

1.如果两个或更多的线程同时尝试获取轻量级锁,并且自旋等待(忙等待)超过了预设的次数,或者有一个线程已经持有锁而其他线程在等待,轻量级锁就会膨胀为重量级锁。
2.在这个过程中,对象头会发生变化,不再直接指向线程的锁记录,而是指向一个Monitor对象。
Monitor对象包含了等待队列(EntryList)和持有锁的线程(owner)等信息。
3.未能获取锁的线程会被放入Monitor的等待队列中,并阻塞等待锁的释放。
4.当持有锁的线程释放锁时,它会通过Monitor的机制来唤醒等待队列中的一个线程,该线程随后会尝试获取锁并继续执行。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/471220.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Python爬虫学习

1.1搭建爬虫程序开发环境 爬取未来七天天气预报 from bs4 import BeautifulSoup from bs4 import UnicodeDammit import urllib.request url"http://www.weather.com.cn/weather/101120901.shtml" try:headers{"User-Agent":"Mozilla/5.0 (Windows …

猫头虎分享已解决Bug ‍ || Go Error: no Go files in /path/to/directory

博主猫头虎的技术世界 🌟 欢迎来到猫头虎的博客 — 探索技术的无限可能! 专栏链接: 🔗 精选专栏: 《面试题大全》 — 面试准备的宝典!《IDEA开发秘籍》 — 提升你的IDEA技能!《100天精通鸿蒙》 …

Fluke ADPT 连接器新增对福禄克万用 Fluke 15B Max 的支持

所需设备: 1、Fluke ADPT连接器; 2、Fluke 15B Max; Fluke 15B Max拆机图: 显示界面如下图: 并且可以将波形导出到EXCEL: 福禄克万用表需要自己动手改造!!!

【JAVA-Day81】 线程休眠: Java 中暂停线程执行的方法 ⏸️

线程休眠: Java 中暂停线程执行的方法 ⏸️💤 线程休眠: Java 中暂停线程执行的方法 ⏸️💤摘要 📝引言 🚀正文 📚一、什么是线程休眠 ⏸️二、线程什么情况下会休眠 ❓三、模拟线程休眠 &#…

电脑重装系统之Windows 10 企业版 LTSC 2021

简介 Windows 10 22H2对于我来说太不简洁,最受不了的一点是微软强行硬塞给我一些并没有什么luan用的应用和功能,比如:天气,Onedrive......以及臃肿的ui设计。而且强行进行自动更新,我是真的受不了这个,看着…

OpenCV-40 绘制直方图

一、使用matplotlib画直方图 可以利用matplotlib把OpenCV统计得到的直方图绘制出来 示例代码如下: import cv2 import matplotlib.pyplot as pltlena cv2.imread("beautiful women.png") # 变为黑白图片 gray cv2.cvtColor(lena, cv2.COLOR_BGR2GRAY…

模型 IPO(输入、处理、输出)学习模型

系列文章 分享 模型,了解更多👉 模型_总纲目录。重在提升认知。信息转化与传递。 1 模型 IPO(输入、处理、输出)学习模型的应用 1.1 项目管理知识体系 PMBOK 中的IPO应用 在项目管理领域,PMBOK(Project Management Body of Know…

【分享】图解ADS+JLINK调试ARM

文章是对LPC2148而写的,但是对三星的44B0芯片同样适用,只需要在选择时将相应的CPU选择的S3C44B0就可以了。 JLINK在ADS下调试心得 前两天一个客户用jlink在ADS下调试LPC2148总报错,这个错误我之前在调试LPC2200的时候也碰到过,后…

数据结构——顺序表专题

目录 1. 数据结构的相关概念什么是数据结构为什么需要数据结构? 2. 顺序表顺序表的概念及结构顺序表分类静态顺序表动态顺序表 3. 动态顺序表的实现准备工作顺序表的初始化顺序表的扩容尾插头插尾删头删指定位置插入数据指定位置删除数据在顺序表中查找销毁 4. 全部…

【C++】实现Date类的各种运算符重载

上一篇文章只实现了operator操作符重载&#xff0c;由于运算符较多&#xff0c;该篇文章单独实现剩余所有的运算符重载。继续以Date类为例&#xff0c;实现运算符重载&#xff1a; 1.Date.h #pragma once#include <iostream> #include <assert.h>using namespace …

Python是垃圾?千万不要再学Python了?

“人生苦短&#xff0c;快学Python”这句话&#xff0c;相信大家都有看到过&#xff0c;但是有细心留意过&#xff0c;就会发现Python其实在网上的评价褒贬不一&#xff0c;有好评&#xff0c;也有差评。这就会给那些不懂Python却想要学Python的一些人造成困惑&#xff0c;我到…

六、Mybatis注解开发

1.MyBatis的常用注解 注解开发越来越流行&#xff0c; Mybatis也可以使用注解开发方式&#xff0c;这样就可以减少编写Mapper映射文件。Insert&#xff1a;实现新增Update&#xff1a;实现更新Delete&#xff1a;实现删除Select&#xff1a;实现查询Result&#xff1a;实现结果…