java 锁

news/2024/11/28 11:51:19/

目录

一,对象头

1,Mark Word

2,指向类的指针

3,数组长度

二,实例数据

三,对齐填充字节

Java对象保存在内存中时,由以下三部分组成:

1,对象头

2,实例数据

3,对齐填充字节

一,对象头
java的对象头由以下三部分组成:

1,Mark Word

2,指向类的指针

3,数组长度(只有数组对象才有)

 

1,Mark Word
Mark Word记录了对象和锁有关的信息,当这个对象被synchronized关键字当成同步锁时,围绕这个锁的一系列操作都和Mark Word有关。

Mark Word在32位JVM中的长度是32bit,在64位JVM中长度是64bit。

Mark Word在不同的锁状态下存储的内容不同,在32位JVM中是这么存的:

锁状态

25bit

4bit

1bit

2bit

23bit

2bit

是否偏向锁

锁标志位

无锁

对象的HashCode

分代年龄

0

01

偏向锁

线程ID

Epoch

分代年龄

1

01

轻量级锁

指向栈中锁记录的指针

00

重量级锁

指向重量级锁的指针

10

GC标记

11

其中无锁和偏向锁的锁标志位都是01,只是在前面的1bit区分了这是无锁状态还是偏向锁状态。

JDK1.6以后的版本在处理同步锁时存在锁升级的概念,JVM对于同步锁的处理是从偏向锁开始的,随着竞争越来越激烈,处理方式从偏向锁升级到轻量级锁,最终升级到重量级锁。

 

JVM一般是这样使用锁和Mark Word的:

1,当没有被当成锁时,这就是一个普通的对象,Mark Word记录对象的HashCode,锁标志位是01,是否偏向锁那一位是0。

2,当对象被当做同步锁并有一个线程A抢到了锁时,锁标志位还是01,但是否偏向锁那一位改成1,前23bit记录抢到锁的线程id,表示进入偏向锁状态。

3,当线程A再次试图来获得锁时,JVM发现同步锁对象的标志位是01,是否偏向锁是1,也就是偏向状态,Mark Word中记录的线程id就是线程A自己的id,表示线程A已经获得了这个偏向锁,可以执行同步锁的代码。

4,当线程B试图获得这个锁时,JVM发现同步锁处于偏向状态,但是Mark Word中的线程id记录的不是B,那么线程B会先用CAS操作试图获得锁,这里的获得锁操作是有可能成功的,因为线程A一般不会自动释放偏向锁。如果抢锁成功,就把Mark Word里的线程id改为线程B的id,代表线程B获得了这个偏向锁,可以执行同步锁代码。如果抢锁失败,则继续执行步骤5。

5,偏向锁状态抢锁失败,代表当前锁有一定的竞争,偏向锁将升级为轻量级锁。JVM会在当前线程的线程栈中开辟一块单独的空间,里面保存指向对象锁Mark Word的指针,同时在对象锁Mark Word中保存指向这片空间的指针。上述两个保存操作都是CAS操作,如果保存成功,代表线程抢到了同步锁,就把Mark Word中的锁标志位改成00,可以执行同步锁代码。如果保存失败,表示抢锁失败,竞争太激烈,继续执行步骤6。

6,轻量级锁抢锁失败,JVM会使用自旋锁,自旋锁不是一个锁状态,只是代表不断的重试,尝试抢锁。从JDK1.7开始,自旋锁默认启用,自旋次数由JVM决定。如果抢锁成功则执行同步锁代码,如果失败则继续执行步骤7。

7,自旋锁重试之后如果抢锁依然失败,同步锁会升级至重量级锁,锁标志位改为10。在这个状态下,未抢到锁的线程都会被阻塞。

 

2,指向类的指针
该指针在32位JVM中的长度是32bit,在64位JVM中长度是64bit。

Java对象的类数据保存在方法区。

 

3,数组长度
只有数组对象保存了这部分数据。

该数据在32位和64位JVM中长度都是32bit。

 

二,实例数据
对象的实例数据就是在java代码中能看到的属性和他们的值。

 

三,对齐填充字节
因为JVM要求java的对象占的内存大小应该是8bit的倍数,所以后面有几个字节用于把对象的大小补齐至8bit的倍数,没有特别的功能。

 

以上。
--------------------- 
作者:lkforce 
来源:CSDN 
原文:https://blog.csdn.net/lkforce/article/details/81128115 
版权声明:本文为博主原创文章,转载请附上博文链接!


http://www.ppmy.cn/news/360288.html

相关文章

[Java]锁

锁 GO! 乐观锁悲观锁自旋锁原理自旋锁优缺点优点缺点自旋锁时间阈值(1.6 引入了适应性自旋锁)自旋锁的开启 Synchronized 同步锁Synchronized作用范围Synchronized核心组件Synchronized实现JDK1.6后的优化 ReentrantLockLock接口主要方法tryLock 和 lock 和 lockInt…

java锁

1.悲观锁 :锁住资源其他线程不能在访问 2.乐观锁: 不锁资源,默认其他进程不会修改资源 锁住同步资源失败 线程是否需要阻塞? 阻塞 不阻塞: 自旋锁 适应性自旋锁 减少了线程上下文的其二环 因为是不阻塞 而…

freeswitch如何判断挂机方

概述 freeswitch作为VOIP的软交换平台,需要对呼叫的信息做判断和归类。 常见的呼叫信息中,挂机方向的信息对于话单统计有很大的用处。 但是fs的原始话单和日志中并没有挂机方向的信息。 环境 centos:CentOS7 freeswitch:v1.…

挂机宝搭建教程

nokvm主控面板推荐操作系统版本 Centos7.6.1810 (纯净的操作系统,无其他软件环境) 主控面板硬件要求配置: - 最低要求 推荐配置 内存 2G 2G CPU - - 带宽 不低于2M 2M 磁盘 / 分区不少于100GB 100GB 主控面板安装&#xff1…

vc编写挂机锁

凡是去过网吧的朋友,对挂机锁一定不陌生,鼠标被限定,键盘关键按键被屏蔽。那么这是如何实现的呢,我们也来写一个吧。 // lockDlg.cpp : implementation file//#include "stdafx.h"#include "lock.h"#include…

关于电脑锁机

电脑锁机的种类 电脑锁机众所周知分为三种 1. 屏幕锁 2. 用户锁 3. 硬盘锁 首先屏幕锁 原理:软件打开后置于最上层并覆盖其他软件界面,通常是全屏,这种锁机比较有威慑性因为界面可以做的比较震撼人心,通常会把一些可以关闭窗口的东西给禁用…

分布式锁解决方案_Redis实现的分布式锁原理

获取锁 互斥:确保只有一个线程获得锁 添加锁 利用setnx的互斥性 启动Redis的我们进行测试看看怎么能某个数据来获得锁 setnx命令表示上锁只要是aa被创建出来我们不删除aa或者设定时间自动删除,那么这把锁就不能被释放开 释放锁 手动释放锁 如果我们…

从CAP角度分析redis分布式锁和zookeeper锁区别

前言: 在分布式系统中,一致性( c )意味着每次读取都会获取最新的写入数据或异常,可用性( a )意味着可以对每个请求得到非异常的响应,而不保证获取最新的写入数据。 分区容错( p )意味着即使节点之间的网络异常,系统也…