您的位置:首页 > 新闻 > 热点要闻 > 国内购物网站大全_私人app一键制作器软件_如何制作网站赚钱_热点新闻最新消息

国内购物网站大全_私人app一键制作器软件_如何制作网站赚钱_热点新闻最新消息

2025/1/11 13:57:27 来源:https://blog.csdn.net/LG_971124/article/details/144377121  浏览:    关键词:国内购物网站大全_私人app一键制作器软件_如何制作网站赚钱_热点新闻最新消息
国内购物网站大全_私人app一键制作器软件_如何制作网站赚钱_热点新闻最新消息

1.乐观锁和悲观锁及使用场景

悲观锁:认为每次操作都会出现线程并发问题,每次都会加锁

乐观锁:认为大部分操作都不会出现问题,通过compare and swap操作来实现锁的效果

在java中悲观锁通过synchronize和lock等来实现

在java中乐观锁通过atomic为前缀的类实现, Atomic 原子操作类是基于CAS + volatile 实现的,并且类中的所有方法都使用 final 修饰,进一步保证线程安全。而 CAS 算法的具体实现方式在于 Unsafe 类中,Unsafe 类的所有方法都是 native 修饰的,也就是说所有方法都是直接调用操作系统底层资源进行执行相应任务。 通过unsafe中的方法,和volatile修饰的变量,去维护一个int类型的value,类似于版本号,然后循环比较并替换,直到成功。

乐观锁一般用于性能要求高,安全性相对较低的场景,例如查多写少的场景

悲观锁一般用于安全性高,性能要求较低的场景,例如写多查少


2.阻塞锁和自旋锁的理解

阻塞锁:会主动释放cpu资源,等到锁被释放后,再次获得锁才会被调用(不会浪费cpu资源,但是会有线程上下文切换)

自旋锁:不会主动释放cpu资源,一直循环获取锁(浪费cpu资源,但不会有线程的切换)

自适应锁:在自旋锁的基础上,增加了一个自选次数的阈值,当达到阈值后跳出循环,避免了死循环

3.Synchronize的原理

synchronize底层基于jvm的内置锁实现,每个对象都会有个monitor监视器锁,通过进入与退出monitor来实现方法和代码块的同步,monitor底层依赖于操作系统的mutex lock互斥锁来实现,这个锁会在用户态和内核态之间切换,性能比较低。

进出monitor原理

当线程进入monitor监视器时,会查看当前monitor数量是否为0,第一次进入时为0,则将monitor数量加1,将当前线程标记为monitor拥有者,执行业务代码,执行完毕后monitor数量减一,然后判断monitor数量是否为0,为0就退出monitor,当前线程不再是monitor拥有者,其他线程可以获取当前monitor。

当已经有线程进入monitor时,新进入的线程会先判断当前monitor数量是否为0,不为0,然后判断当前线程是否是monitor拥有者,不是拥有者进入阻塞状态,等到锁被释放时,被唤醒。

同一个线程重入monitor时,会先判断当前monitor是否为0,不为0则判断当前线程是否是monitor拥有者,是拥有者则monitor数量加一,然后执行业务代码,执行完毕后monitor数量减一,然后判断当前monitor数量是否为0,不为0则结束,直到monitor为0,才会释放锁。

锁升级原理

依次是无锁-》偏向锁-》轻量级锁-》重量级锁

每个对象都拥有对象头,对象头由Mark World ,指向类的指针,以及数组长度三部分组成,锁升级主要依赖Mark Word锁标记为中的锁标志位和释放偏向锁标识位。

  • Mark Word锁标记为偏向锁时,Mark Word存储的是当前拥有偏向锁的线程的线程id
  • Mark Word锁标记为轻量级锁时,Mark Word存储的是当前拥有锁线程栈中的LockRecord指针
  • Mark Word锁标记为重量级锁时, 为指向堆中的monitor对象的指针。

锁升级过程

  1. 当线程进入同步代码块时,检查锁对象Mark World中的锁状态,如果为无锁状态,通过CAS操作将锁对象头中的锁状态改为偏向锁,并将当前线程的线程id记录到锁对象Mark World中,并拷贝到自己对象头中。
  2. 当再次进入同步代码块时,发现当前锁状态为偏向锁,判断锁对象Mark World中的线程id是否相等,如果相等就获取到锁。如果不相等,则通过cas操作替换锁对象头中线程id,更改成功获取到锁。更改失败则表示有竞争。偏向锁升级为轻量级锁,当持有偏向锁线程到达全局安全(safepoint)时,获得偏向锁的线程被挂起,检查当前持有锁线程状态,已经执行完毕,将持有锁线程markworld锁偏向位改为无锁。
  3. 如果还没有执行完毕, 在持有偏向锁线程的栈帧中建立一个名为锁记录(Lock Record)的空间,拷贝锁对象头中的Mark Word到原持有偏向锁线程的锁记录中,原持有偏向锁线程获得轻量级锁,将锁对象头中Mark Word指向原持有偏向锁线程的LockRecord指针, 并将Lock record里的owner指针指向锁对象头的mark word,然后被阻塞在安全点的线程继续往下执行同步代码。
  4. 在当前线程的栈帧中建立锁记录(Lock Record)的空间 ,当前线程拷贝对象头中的Mark Word到自己锁记录中,通过CAS操作尝试将锁对象头中Mark Word指向自己LockRecord指针。成功,获取到轻量级锁,失败,虚拟机首先会检查锁对象的Mark Word是否指向当前线程的栈帧,如果是就说明当前线程已经拥有了这个对象的锁,现在是重入状态,那么设置Lock Record第一部分(Displaced Mark Word)为null,起到了一个重入计数器的作用。如果不是说明这个锁对象已经被其他线程抢占了,说明此时有多个线程竞争锁,那么它就会自旋等待锁,一定次数后仍未获得锁对象,说明发生了竞争,需要膨胀为重量级锁。
  5. 重量级锁将锁对象头中Mark Word指向堆中monitor对象。

偏向锁(无锁)

大多数情况下锁不仅不存在多线程竞争,而且总是由同一线程多次获得。偏向锁的目的是在某个线程 获得锁之后(线程的id会记录在对象的Mark Word锁标志位中),消除这个线程锁重入(CAS)的开销,看起来让这个线程得到了偏护。(第二次还是这个线程进来就不需要重复加锁,基本无开销),如果自始至终使用锁的线程只有一个,很明显偏向锁几乎没有额外开销,性能极高。

偏向锁释放过程:

偏向锁只有遇到其他线程尝试竞争偏向锁时,持有偏向锁的线程才会释放锁,线程不会主动去释放偏向锁。偏向锁的撤销,需要等待全局安全点safepoint,它会首先暂停拥有偏向锁的线程A,然后判断这个线程A,此时有两种情况:

A 线程已经退出了同步代码块,或者是已经不再存活了,此时就会直接撤销偏向锁,变成无锁状态。

A 线程还在同步代码块中,此时将A线程的偏向锁升级为轻量级锁。

轻量级锁(CAS):

轻量级锁是由偏向锁升级来的,偏向锁运行在一个线程进入同步块的情况下,当第二个线程加入锁争用的时候,偏向锁就会升级为轻量级锁自旋锁);没有抢到锁的线程将自旋,获取锁的操作。轻量级锁的意图是在没有多线程竞争的情况下,通过CAS操作尝试将MarkWord锁标志位更新为指向LockRecord的指针,减少了使用重量级锁的系统互斥量产生的性能消耗。

长时间的自旋操作是非常消耗资源的,一个线程持有锁,其他线程就只能在原地空耗CPU,执行不了任何有效的任务,这种现象叫做忙等(busy-waiting)

批量重偏向

当只有一个线程反复进入同步块时,偏向锁带来的性能开销基本可以忽略,但是当有其他线程尝试获得锁时,就需要等到safe point时将偏向锁撤销为无锁状态或升级为轻量级/重量级锁。这个过程是要消耗一定的成本的,所以如果说运行时的场景本身存在多线程竞争的,那偏向锁的存在不仅不能提高性能,而且会导致性能下降。因此,JVM中增加了一种批量重偏向/撤销的机制。

批量重偏向的原理

  1. 首先引入一个概念epoch,其本质是一个时间戳,代表了偏向锁的有效性,epoch存储在可偏向对象的MarkWord中。除了对象中的epoch,对象所属的类class信息中,也会保存一个epoch值。
  2. 每当遇到一个全局安全点时(这里的意思是说批量重偏向没有完全替代了全局安全点,全局安全点是一直存在的),比如要对class C进行批量再偏向,则首先对 class C中保存的epoch进行增加操作,得到一个新的epoch_new
  3. 然后扫描所有持有 class C 实例的线程栈,根据线程栈的信息判断出该线程是否锁定了该对象,仅将epoch_new的值赋给被锁定的对象中,也就是现在偏向锁还在被使用的对象才会被赋值epoch_new。
  4. 退出安全点后,当有线程需要尝试获取偏向锁时,直接检查 class C 中存储的 epoch 值是否与目标对象中存储的 epoch 值相等,如果不相等,则说明该对象的偏向锁已经无效了(因为上一步里面已经说了只有偏向锁还在被使用的对象才会有epoch_new,这里不相等的原因是class C里面的epoch值是epoch_new,而当前对象的epoch里面的值还是epoch),此时竞争线程可以尝试对此对象重新进行偏向操作,无需再等待到安全点检查。

重量级锁:

如果锁竞争情况严重,某个达到最大自旋次数(10次默认)的线程,会将轻量级锁升级为重量级锁,重量级锁则直接将自己挂起,在JDK1.6之前,synchronized直接加重量级锁,很明显现在得到了很好的优化。

虚拟机使用CAS操作尝试将MarkWord更新为指向LockRecord的指针,如果更新成功表示线程就拥有该对象的锁;如果失败,会检查MarkWord是否指向当前线程的栈帧,如果是,表示当前线程已经拥有这个锁;如果不是,说明这个锁被其他线程抢占,此时膨胀为重量级锁。

4.说一下Synchronize与lock的区别

  1. synchronize是关键字,lock是接口;
  2. synchornize底层基于jvm内置锁,无需管理。lock需要手动加锁,并且需要手动释放锁,一般在finally中释放。
  3. synchornize是可重入,非公平,不可判断锁;lock是根据实现类手动是否公平,是否重入,可判断锁。
  4. synchornize是需要释放锁后其他线程才能获得锁,lock根据锁的方式例如共享锁,即是其他没有释放锁也可以共享

5.lock锁底层是基于什么实现的

lock锁是juc包下locks包下的接口,java的锁基本都是lock的实现类,底层实现基于aqs;

aqs通过维护一个volatile修饰的int类型的 state 和一个fifo的双向链表来实现,每个node节点维护pre和next指针。以ReentrantLock为例,state初始化为0,表示未锁定状态。当线程获取锁时,会独占该锁并将stater+1。此后,其他线程再获取锁就会失败,加入到node节点中,直到线程释放锁,stater=0为止,如果是公平锁则是next指向的node节点获取锁,非公平锁则会出现插队,直接获取锁。线程释放锁之前,可以重复获得锁,stater会累加,达到重入锁的效果,重入多少次,就会释放多次,保证state=0为止。

6.介绍一下mysql中的锁

  • 是否加锁分为
    • 乐观锁:通过新增一个字段版本号version,通过cas操作来实现锁的效果
    • 悲观锁:通过在sql语句后加for update
  • 锁属性分为:
    • 共享锁:读锁,即可以查询同一条数据,不能写(lock in share mode)
    • 排它锁:写锁,不能写不能查
  • 锁粒度分为:
    • 行锁:锁住的是表的某一行或多行记录,只有锁住的数据不能访问,粒度小不容易冲突,加锁麻烦
      • 记录锁:锁的范围是表中的某一行记录,避免脏读问题
      • 间隙锁:锁住的是表记录的某一个区间,当表相邻id之间的空隙形成一个区间,遵循左开右开的原则,间隙锁只会出现在重复读的事务级别中。同时解决了幻读问题。
      • 临间锁:记录锁和间隙锁的结合,遵循左开右闭的基本原则,同时也可以锁单行
    • 表锁:锁住整个表
      • 自增锁:特殊的表级锁,对于拥有自增主键的表,保证自增主键的唯一性的锁机制
    • 页锁:介于行锁与表锁之间,按页加锁
    • 全局锁:锁定数据库的所有表

7.什么是共享锁和排它锁

共享锁:也叫读锁,可以查询同一个数据,但不能进行写操作

排它锁:也叫写锁,同一数据,不能查询也不能进行写操作

8.间隙锁和临键锁额区别

间隙锁:采取左开右开的原则,只包含查询的间隙

临键锁:间隙锁+记录锁,不仅采取了左闭右开,同时还可以锁查询到的行记录

9.mysql事务的四大特性,原子性和持久性的原理

事务四大特性(ACID):原子性,一致性,持久性,隔离性

原子性:把一组事务操作看成一个整体,这一组操作要么同时成功要么同时失败

一致性:数据改变前后的总量保持不变

持久性:事务一旦提交,就会持久化到磁盘中

隔离性:事务之间相互隔离,根据隔离级别不同会有不同的影响

原子性的原理:依赖于undolog日志,在事务开始前会先将当前数据存入undolog日志,事务执行失败或回滚根据undolog日志进行回滚。

持久性原理:mysql持久性基于redolog日志实现。redolog是新数据的备份,在事务提交之前,会先写redolog日志,并将redolog日志持久化,此时宕机,就可以根据redolog日志进行恢复。

10.mysql的事务执行原理是怎样的

mysql事务执行开始后,会先将操作的数据写入到undolog日志中,然后进行操作,操作完成将数据写入到redolog日志中,并将redolog日志持久化到磁盘,然后将事务提交

11.事务的隔离级别有哪些,分别解决什么问题

事务的隔离级别分为:读未提交,读已提交,可重读,串行化

读未提交:是最低级别,会出现脏读幻读重读等问题

读已提交:解决了脏读的问题,脏读就是读取到了并行事务中事务还没有提交的数据

可重读:解决了不可重复读的问题,同时在可重读的等级中,mysql通过间隙锁解决了幻读的问题。

不可重复读问题:就是在两个并行的事务中,A事务中查询数据,并进行修改,B事务中查询多次同样的数据,由于A修改了数据,导致多次查询结果不相同。

幻读:在两个并行事务中,A事务中查询数据,并新增了一条数据。B事务中查询多次同样的数据,但是由于A事务中插入了新的数据,导致多次查询的结果条数不一致。

串行化:类似于单线程,将事务按顺序一个一个执行,可以解决上诉所有问题,但是性能低

12.mysql事务隔离级别的实现原理是怎样的

事务隔离级别利用MVCC机制(多版本并发控制)实现,就是为一个行记录数据生成多个版本的数据快照,这些快照在undolog日志中,如果当前行操作正在进行写操作,那么读操作不会被阻塞而读取的是该行的数据快照。

13.读已提交和可重复读在生成版本快照时有什么区别

读已提交:在每次读取数据的时候都会为该行生成一个新的版本快照,因此会出现不可重复读的问题

可重复:在第一次查询数据的时候会生成数据快照,后续查询都使用同一个快照,解决了不可重复读的问题,但会造成第二类更新丢失的问题,即两组并行事务中,查询了同一条数据,都在原数据基础上执行了更改,那么最先提交的事务更新会被覆盖,可以通过乐观锁和悲观锁解决。

14.什么是分布式锁

分布式锁是一种在分布式系统环境下实现的锁机制,它主要用于解决,多个分布式节点之间对共享资源的互斥访问问题,确保在分布式系统中,即使存在有多个不同节点上的进程或线程,同一时刻也只有一个节点可以获得锁并对共享资源进行操作,从而维护数据的一致性和完整性。

15.分布式锁有哪些解决方案

  1. 基于数据库操作(主键或唯一索引)
    1. 通过主键或唯一索引,通过锁住当前操作数据的主键,来实现
  1. 基于redis
    1. 通过setnx指令来加锁,会出现死锁问题,因此需要设置锁过期时间,同时需要保证锁过期时间与setnx之间的原子性,可以通过set nx ex来实现。并且在删除锁是需要判断当前锁是否为当前线程所拥有,避免误删除,可以通过加一个全局唯一的标识,每次删除前判断,并且要保证判断与删除的原子性,可以通过lua脚本来实现。
    2. 使用redis已经封装好的redission来实现分布式锁,redission中有一个看门狗机制,当我们没有设置过期时间的时候,看门狗会每10秒检查当前锁,如果还在操作则延长过期时间,每次延长30秒。如果设置了过期时间,则看门狗机制失效
    3. redission中的锁分为:
      1. 可重入锁;reentrantLock,同一个线程重复获取锁
      2. 公平锁:按照请求顺序获取锁
      3. 读写锁:共享锁和排他锁
      4. 红锁:解决集群模式中脑裂问题,需要一半以上主节点加锁成功才能获取到锁
      5. 联锁:将多个锁连到一起,获取到所有锁,才能获取到真正的锁
      6. 闭锁:多个任务,都可以看到执行的结果,当都完成了,才释放锁
      7. 信号量:可以维护一个整数,通常用来做数据的记录
      8. 可过期信号量:在信号量的基础上,多了一个过期时间
  1. 基于zookeeper临时顺序节点+watch
    1. 临时节点:临时节点唯一且会自动过期,当新增节点成功则获取到锁
    2. 临时顺序节点+watch:临时顺序节点就是每个请求进来都按顺序建新的节点,当第一个节点执行完成,后会删除节点,下一个节点会监听上一个节点的删除事件,如果是公平锁,则下一个节点获取到锁

16.Redis如何实现分布式锁,用什么命令

通过setnx指令来加锁,会出现死锁问题,因此需要设置锁过期时间,同时需要保证锁过期时间与setnx之间的原子性,可以通过set nx ex来实现。并且在删除锁是需要判断当前锁是否为当前线程所拥有,避免误删除,可以通过加一个全局唯一的标识,每次删除前判断,并且要保证判断与删除的原子性,可以通过lua脚本来实现。

17.Redission的看门狗原理吗

redission中有一个看门狗机制,当我们没有设置过期时间的时候,看门狗会每10秒检查当前锁,如果还在操作则延长过期时间,每次延长30秒。如果设置了过期时间,则看门狗机制失效

18.Redission的锁有哪些 分别有什么作用

  1. 可重入锁;reentrantLock,同一个线程重复获取锁
  2. 公平锁:按照请求顺序获取锁
  3. 读写锁:共享锁和排他锁
  4. 红锁:解决集群模式中脑裂问题,需要一半以上主节点加锁成功才能获取到锁
  5. 联锁:将多个锁连到一起,获取到所有锁,才能获取到真正的锁
  6. 闭锁:多个任务,都可以看到执行的结果,当都完成了,才释放锁
  7. 信号量:可以维护一个整数,通常用来做数据的记录
  8. 可过期信号量:在信号量的基础上,多了一个过期时间


19.如何使用ZK实现分布式锁的

  1. 临时节点:临时节点唯一且会自动过期,当新增节点成功则获取到锁
  2. 临时顺序节点+watch:临时顺序节点就是每个请求进来都按顺序建新的节点,当第一个节点执行完成,后会删除节点,下一个节点会监听上一个节点的删除事件,如果是公平锁,则下一个节点获取到锁

版权声明:

本网仅为发布的内容提供存储空间,不对发表、转载的内容提供任何形式的保证。凡本网注明“来源:XXX网络”的作品,均转载自其它媒体,著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处。

我们尊重并感谢每一位作者,均已注明文章来源和作者。如因作品内容、版权或其它问题,请及时与我们联系,联系邮箱:809451989@qq.com,投稿邮箱:809451989@qq.com