Java中的synchronized关键字

 更新时间:2021年12月12日 11:02:25   作者:bkpp976  
这篇文章主要介绍了Java中的synchronized关键字,synchronized可以保证方法或代码块在运行时,同一时刻只有一个线程可以进入到临界区(互斥性),同时它还保证了共享变量的内存可见性,下面我们就来看看你文章对synchronized锁的介绍,需要的朋友也可以参考一下

1、synchronized锁的底层实现原理

JVM基于进入和退出Monitor对象来实现方法同步和代码块同步。代码块同步是使用monitorentermonitorexit指令实现的,monitorenter指令是在编译后插入到同步代码块的开始位置,而monitorexit是插入到方法结束处和异常处。任何对象都有一个monitor与之关联,当且一个monitor被持有后,它将处于锁定状态。

根据虚拟机规范的要求,在执行monitorenter指令时,首先要去尝试获取对象的锁,如果这个对象没被锁定,或者当前线程已经拥有了那个对象的锁,把锁的计数器加1;相应地,在执行monitorexit指令时会将锁计数器减1,当计数器被减到0时,锁就释放了。如果获取对象锁失败了,那当前线程就要阻塞等待,直到对象锁被另一个线程释放为止。

如何判断这个对象是否被锁定?对象头中的MarkWord字段记录了该对象的锁信息。

2、基于synchronized实现单例模式

public class Singleton {
    private volatile static Singleton uniqueInstance;

    private Singleton(){

    }
    public static Singleton getUniqueInstance() {
        //没有实例化才加锁
        if (uniqueInstance == null) {
            //给类对象加锁
            synchronized (Singleton.class) {
                if (uniqueInstance == null)
                    uniqueInstance = new Singleton();
            }
        }
        return uniqueInstance;
    }

//    public static synchronized Singleton getUniqueInstance(){
//        if(uniqueInstance==null){
//            uniqueInstance = new Singleton();
//        }
//        return uniqueInstance;
//    }
}

首先说一下为什么不采用第二种方式实现单例:不管该对象是否已经实例化,都要调用这个同步方法,会导致大量的线程进入阻塞;而采用双重锁检验,可以在第一次判断不为空的时候就直接返回,不用进入同步代码块。

几个要点:

  • 为什么uniqueInstance属性要用volatile修饰?new操作并非一个原子性操作,分为三个步骤(分配对象的内存空间、初始化对象、设置uniqueInstance指向刚分配的内存地址),如果不使用volatile,2和3之间可能发生指令重排,导致外部访问到一个还没有初始化的对象。
  • 为什么构造方法时私有的?防止对象在其他地方被创建。
  • 为什么uniqueInstance是私有静态的?私有使得外部只能通过特定的方式去访问对象,静态是因为要在静态方法中访问该对象。
  • 为什么getUniqueInstance()方式是公共、静态的?public使得外界统一通过访问该方法获得对象,static使得程序可以通过类名获取对象。
  • 为什么采用双重检测初始化对象?第一次检测主要用来判断对象是否已经创建,如果已创建则直接返回;第二次检测是因为:可能有多个线程在第一次检测中发现对象为空,同时进入同步代码块,但只有一个线程会抢到锁并创建对象,其他线程阻塞排队等待锁的释放。当创建对象的线程返回后,阻塞的线程会被唤醒,这时候对象已经不为空,所以需要第二次检测来阻止对象的多次创建。

3、利用类加载实现单例模式(饿汉模式)

public class SingleTon2 {
/** 内置对象是静态的,并且直接创建对象,保证对象在初始化时加载完成 */
    public static SingleTon2 instance = new SingleTon2(); 

/** 构造方法私有,防止对象在其他地方被创建 */
    private SingleTon2(){

    }

/** 公共静态方法返回对象 */
    public static SingleTon2 getInstance(){
        
        return instance;
    }
}

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

相关文章

  • Java 在 Word 文档中使用新文本替换指定文本的方法

    Java 在 Word 文档中使用新文本替换指定文本的方法

    这篇文章主要介绍了Java 在 Word 文档中使用新文本替换指定文本的方法,本文通过实例代码给大家介绍的非常详细,具有一定的参考借鉴价值,需要的朋友可以参考下
    2019-09-09
  • java实现两张图片2D翻转动画效果

    java实现两张图片2D翻转动画效果

    这篇文章主要为大家详细介绍了java实现两张图片2D翻转动画效果,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-08-08
  • SpringBoot项目整合jasypt实现过程详解

    SpringBoot项目整合jasypt实现过程详解

    这篇文章主要介绍了SpringBoot项目整合jasypt实现过程详解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-08-08
  • Java并发编程中的生产者与消费者模型简述

    Java并发编程中的生产者与消费者模型简述

    这篇文章主要介绍了Java并发编程中的生产者与消费者模型简述,多线程并发是Java编程中最终要的部分之一,需要的朋友可以参考下
    2015-07-07
  • Java-Redis-Redisson分布式锁的功能使用及实现

    Java-Redis-Redisson分布式锁的功能使用及实现

    这篇文章主要介绍了Java-Redis-Redisson-分布式锁的功能使用及实现,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2022-08-08
  • IntelliJ-Idea导出可执行Jar流程解析

    IntelliJ-Idea导出可执行Jar流程解析

    这篇文章主要介绍了IntelliJ-Idea导出可执行Jar流程解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-12-12
  • Java8新特性之Optional使用详解

    Java8新特性之Optional使用详解

    这篇文章主要介绍了Java8新特性之Optional使用详解,为了解决空指针异常更加优雅,Java8 提供了 Optional 类库,Optional 实际上是个容器,它可以保存类型T的值,或者仅仅保存null,,需要的朋友可以参考下
    2023-08-08
  • Spring Boot优化后启动速度快到飞起技巧示例

    Spring Boot优化后启动速度快到飞起技巧示例

    这篇文章主要为大家介绍了Spring Boot优化后启动速度快到飞起的技巧示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-07-07
  • springboottest测试依赖和使用方式

    springboottest测试依赖和使用方式

    这篇文章主要介绍了springboottest测试依赖和使用方式,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-03-03
  • java实战技巧之if-else代码优化技巧大全

    java实战技巧之if-else代码优化技巧大全

    代码中如果if-else比较多,阅读起来比较困难,维护起来也比较困难,很容易出bug,下面这篇文章主要给大家介绍了关于java实战技巧之if-else代码优化技巧的相关资料,需要的朋友可以参考下
    2022-02-02

最新评论