Java的synchronized关键字深入解析

 更新时间:2023年12月07日 10:07:16   作者:外星喵  
这篇文章主要介绍了Java的synchronized关键字深入解析,在并发编程中,多线程同时并发访问的资源叫做临界资源,当多个线程同时访问对象并要求操作相同资源时,分割了原子操作就有可能出现数据的不一致或数据不完整的情况,需要的朋友可以参考下

概念

在并发编程中,多线程同时并发访问的资源叫做临界资源,当多个线程同时访问对象并要求操作相同资源时,分割了原子操作就有可能出现数据的不一致或数据不完整的情况,为避免这种情况的发生,我们会采取同步机制,以确保在某一时刻,方法内只允许有一个线程。

sychronized 用于 修饰 代码块、类的实例方法 & 静态方法

是利用锁的机制来实现同步的。

锁机制有如下两种特性:

**互斥性:**即在同一时间只允许一个线程持有某个对象锁,通过这种特性来实现多线程中的协调机制,这样在同一时间只有一个线程对需同步的代码块(复合操作)进行访问。互斥性我们也往往称为操作的原子性。

**可见性:**必须确保在锁被释放之前,对共享变量所做的修改,对于随后获得该锁的另一个线程是可见的(即在获得锁时应获得最新共享变量的值),否则另一个线程可能是在本地缓存的某个副本上继续操作从而引起不一致。

synchronized的用法

在这里插入图片描述

代码如下:

/**
* 对象锁
*/
public class Test{ 
    // 对象锁:形式1(方法锁) 
    public synchronized void Method1(){ 
        System.out.println("我是对象锁也是方法锁"); 
        try{ 
            Thread.sleep(500); 
        } catch (InterruptedException e){ 
            e.printStackTrace(); 
        } 
    } 
    // 对象锁:形式2(代码块形式) 
    public void Method2(){ 
        synchronized (this){ 
            System.out.println("我是对象锁"); 
            try{ 
                Thread.sleep(500); 
            } catch (InterruptedException e){ 
                e.printStackTrace(); 
            } 
        } 
    } 
/**
 * 方法锁(即对象锁中的形式1)
 */
    public synchronized void Method1(){ 
        System.out.println("我是对象锁也是方法锁"); 
        try{ 
            Thread.sleep(500); 
        } catch (InterruptedException e){ 
            e.printStackTrace(); 
        } 
    } 
/**
 * 类锁
 */
   // 类锁:形式1 :锁静态方法
    public static synchronized void Method1(){ 
        System.out.println("锁静态方法"); 
        try{ 
            Thread.sleep(500); 
        } catch (InterruptedException e){ 
            e.printStackTrace(); 
        } 
    } 
    // 类锁:形式2 :锁静态代码块
    public void Method2(){ 
        synchronized (Test.class){ 
            System.out.println("锁静态代码块"); 
            try{ 
                Thread.sleep(500); 
            } catch (InterruptedException e){ 
                e.printStackTrace(); 
            } 
        } 
    } 
}

使用synchronized注意的问题

  • 与moniter关联的对象不能为空
  • 尽量减小synchronized作用域范围
  • 不能使用不同的monitor锁相同的方法
  • 防止多个锁的交叉导致死锁

synchronized原理

采用 synchronized 修饰符实现的同步机制叫做互斥锁机制,它所获得的锁叫做互斥锁。每个对象都有一个 monitor (锁标记),当线程拥有这个锁标记时才能访问这个资源,没有锁标记便进入锁池。任何一个对象系统都会为其创建一个互斥锁,这个锁是为了分配给线程的,防止打断原子操作。每个对象的锁只能分配给一个线程,因此叫做互斥锁。

在 Java 中,每个对象有一把锁,叫对象锁,针对每个类也有一个锁,可以称为“类锁”,类锁实际上是通过对象锁实现的,即类的 Class 对象锁。每个类只有一个 Class 对象,所以每个类只有一个类锁。

底层依赖于 JVM ,通过一个监视器对象monitor完成, wait、notify 等方法也依赖于 monitor 对象监视器锁(monitor的本质 依赖于 底层操作系统的互斥锁Mutex Lock实现)

在 Java 中,每个对象都会有一个 monitor 对象,监视器。

  • 某一线程占有这个对象的时候,先monitor 的计数器是不是0,如果是0还没有线程占有,这个时候线程占有这个对象,并且对这个对象的monitor+1;如果不为0,表示这个线程已经被其他线程占有,这个线程等待。当线程释放占有权的时候,monitor-1;
  • 同一线程可以对同一对象进行多次加锁,+1,+1,重入性

在这里插入图片描述

我们先看一段简单的代码:

public class TestSynchronized {
    private int count;
    @Test
    public void test() throws InterruptedException {
        synchronized (this){
            count++;
        }
    }
}

现在我们反编译一下:

在这里插入图片描述

通过反编译后的代码我们可以知道synchronized修饰的代码块的加锁其实是通过monitorenter和monitorExit汇编指令配合使用的。在执行 count++ 指令之前,编译器加了一条 monitorenter 指令,count++ 指令执行结束时又加了一条 monitorexit 指令。准确意义上来说,这就是两条加锁的释放锁的指令。

除此之外,我们的 synchronized 方法在反编译后并没有这两条指令,但是编译器却在方法表的 flags 属性中设置了一个标志位 ACC_SYNCHRONIZED。这样,每个线程在调用该方法之前都会检查这个状态位是否为 1,如果状态为 1 说明这是一个同步方法,需要首先执行 monitorenter 指令去尝试获取当前实例对象的内置锁,并在方法执行结束执行 monitorexit 指令去释放锁。

JVM对synchronized的优化

synchronized锁住的代码块:同一时刻只能由一个线程访问,属于悲观锁

对象头与monitor

一个对象实例包含:对象头、实例变量、填充数据

在这里插入图片描述

偏向锁

在对象第一次被某一线程占有的时候,是否偏向锁置1,锁表01,写入线程号,当其他的线 程访问的时候,竞争,失败则升级为轻量级锁

CAS(campany and set

竞争失败的时候,不是马上转化级别,而是执行几次空循环5 10

锁消除

JIT在编译的时候吧不必要的锁去掉

到此这篇关于Java的synchronized关键字深入解析的文章就介绍到这了,更多相关synchronized关键字内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Spring Boot对接Oracle数据库具体流程

    Spring Boot对接Oracle数据库具体流程

    这篇文章主要给大家介绍了关于Spring Boot对接Oracle数据库的具体流程,本文将介绍如何在Spring Boot中连接Oracle数据库的基本配置,包括添加依赖、配置数据源、配置JPA等,需要的朋友可以参考下
    2023-11-11
  • Spring框架中的@Conditional系列注解详解

    Spring框架中的@Conditional系列注解详解

    这篇文章主要介绍了Spring框架中的@Conditional系列注解详解,我们需要一个类实现Spring提供的Condition接口,它会匹配@Conditional所符合的方法,然后我们可以使用我们在@Conditional注解中定义的类来检查,需要的朋友可以参考下
    2024-01-01
  • 一篇文章带你了解JAVA面对对象应用

    一篇文章带你了解JAVA面对对象应用

    Java是一门面向对象的语言。对象是Java程序中的基本实体。除了对象之外Java程序同样处理基本数据。下面这篇文章主要给大家总结了关于Java中面向对象的知识点,需要的朋友可以参考借鉴,下面来一起看看吧
    2021-08-08
  • spring单例如何改多例

    spring单例如何改多例

    这篇文章主要介绍了spring单例如何改多例,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-01-01
  • Java实现获取某年某月第一天/最后一天的方法

    Java实现获取某年某月第一天/最后一天的方法

    这篇文章主要介绍了Java实现获取某年某月第一天/最后一天的方法,涉及java日期运算相关操作技巧,需要的朋友可以参考下
    2018-02-02
  • Java实现纪元秒和本地日期时间互换的方法【经典实例】

    Java实现纪元秒和本地日期时间互换的方法【经典实例】

    这篇文章主要介绍了Java实现纪元秒和本地日期时间互换的方法,结合具体实例形式分析了Java日期时间相关操作技巧,需要的朋友可以参考下
    2017-04-04
  • SpringBoot多数据源配置并通过注解实现动态切换数据源

    SpringBoot多数据源配置并通过注解实现动态切换数据源

    本文主要介绍了SpringBoot多数据源配置并通过注解实现动态切换数据源,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2022-08-08
  • java 多线程交通信号灯模拟过程详解

    java 多线程交通信号灯模拟过程详解

    这篇文章主要介绍了java 多线程交通信号灯模拟过程详解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2019-07-07
  • 详解使用spring aop实现业务层mysql 读写分离

    详解使用spring aop实现业务层mysql 读写分离

    本篇文章主要介绍了使用spring aop实现业务层mysql 读写分离,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-01-01
  • springcloud中RabbitMQ死信队列与延迟交换机实现方法

    springcloud中RabbitMQ死信队列与延迟交换机实现方法

    死信队列是消息队列中非常重要的概念,同时我们需要业务场景中都需要延迟发送的概念,比如12306中的30分钟后未支付订单取消,那么本期,我们就来讲解死信队列,以及如何通过延迟交换机来实现延迟发送的需求,感兴趣的朋友一起看看吧
    2022-05-05

最新评论