synchronized锁及其原理

# Java并发机制的底层实现之volatile关键字 提到,volatile修饰的共享变量并不能具有原子性,对一个变量的修改,乃至一段代码的执行,如果想具有原子性的话,需要使用synchronized关键字进行修饰,即对代码块或者方法进行加锁处理,加锁之后,线程只能按照顺序执行这个代码块或者方法。相当于多车道在前方突然变成了单条车道,这时就得一辆辆车通行了,这无疑会导致系统的执行效率降低,但有时也是无奈之举。

下面的代码演示了synchronized关键字的使用:

java 复制代码
public class SynchronizedTestDemo01 {
    static int number = 0;

    synchronized static void addOne() {
        number++;
    }

    public static void main(String[] args) throws InterruptedException {

        for (int i = 0; i < 10; i++) {
            new Thread(() -> {
                for (int j = 0; j < 1000; j++) {
                    addOne();
                }
            }).start();
        }
        TimeUnit.SECONDS.sleep(5);
        System.out.println(number);
    }
}

这段代码每次输出的都是10000,说明经过synchronized关键字修饰以后,addOne()方法中的代码具有原子性。这里还有另外一个引申出一个问题,这里我的number没有经过volatile关键字修饰,它是怎么做到每一次更新都被其他线程读取到的,是不是synchronized关键字也有volatile关键字的可见性效果?

每次在进入到synchronized关键字修饰的代码块或者方法的时候,都会清空这个线程的CPU缓存,代码块或者方法里面的变量值都要去主内存中获取。每次退出synchronized关键字修饰的代码块或者方法的时候,都会将线程在CPU缓存中的内容刷新到主内存中。所以synchronized关键字确实有volatile关键字的可见性效果。

synchronized关键字是如何实现锁功能的?查看addOne()方法的字节码:

java 复制代码
 static synchronized void addOne();
    descriptor: ()V
    flags: (0x0028) ACC_STATIC, ACC_SYNCHRONIZED
    Code:
      stack=2, locals=0, args_size=0
         0: getstatic     #7                  // Field number:I
         3: iconst_1
         4: iadd
         5: putstatic     #7                  // Field number:I
         8: return
      LineNumberTable:
        line 11: 0
        line 12: 8

可以看到 flags 中有一个ACC_SYNCHRONIZED标识,表示这是一个经过synchronized关键字修饰的方法。

如果使用synchronized关键字修饰代码块,如下所示:

java 复制代码
import java.util.concurrent.TimeUnit;
public class SynchronizedTestDemo02 {
    static int number = 0;

    static void addOne() {
        synchronized (SynchronizedTestDemo02.class){
            number++;
        }
    }
    public static void main(String[] args) throws InterruptedException {

        for (int i = 0; i < 10; i++) {
            new Thread(() -> {
                for (int j = 0; j < 1000; j++) {
                    addOne();
                }
            }).start();
        }
        TimeUnit.SECONDS.sleep(5);
        System.out.println(number);
    }
}

查看addOne()方法的字节码:

java 复制代码
 static void addOne();
    descriptor: ()V
    flags: (0x0008) ACC_STATIC
    Code:
      stack=2, locals=2, args_size=0
         0: ldc           #7                  // class com/example/threadmodule/synchronizedtest/SynchronizedTestDemo02
         2: dup
         3: astore_0
         4: monitorenter
         5: getstatic     #9                  // Field number:I
         8: iconst_1
         9: iadd
        10: putstatic     #9                  // Field number:I
        13: aload_0
        14: monitorexit
        15: goto          23
        18: astore_1
        19: aload_0
        20: monitorexit
        21: aload_1
        22: athrow
        23: return

可以看到字节码的第4行和第14行分别是monitorenter和monitorexit指令,JVM就是靠它两实现的获取锁和释放锁。其实对于ACC_SYNCHRONIZED标识,JVM在执行时,会隐式的调用monitorenter获取锁,当方法退出时或抛出异常时,会隐式的调用monitorexit释放锁。

综上,当JVM遇到monitorenter指令的时候会尝试去获取锁,当遇到monitorexit指令的时候,会从同步代码块中退出,并释放锁。至于多个线程竞争锁的细节,后面将再介绍

相关推荐
wang09074 小时前
自己动手写一个spring之IOC_2
java·后端·spring
来杯@Java5 小时前
学生选课管理系统(基于springboot+vue前后端分离的项目)计算机毕业设计java
java·spring boot·spring·vue·毕业设计·maven·mybatis
不知名的老吴6 小时前
线程的生命周期之线程“插队“
java·开发语言·python
ANnianStriver6 小时前
PetLumina-02-后端开发与前后端联调
java·ai·sa-token
杨了个杨89826 小时前
Keepalived + Nginx + HAProxy 高可用架构部署实战案例
java·nginx·架构
马士兵教育8 小时前
Java还有前景吗?Java+AI大模型学习路线及项目?
java·人工智能·python·学习·机器学习
snow@li9 小时前
Java:理解 Gradle / 后端项目的管家 / 打包SpringBoot 应用 / 完成编译、下载依赖、运行测试、打包 JAR/WAR / 速查表
java
云烟成雨TD9 小时前
Spring AI 1.x 系列【57】动态工具发现:Tool Search Tool
java·人工智能·spring
zfoo-framework9 小时前
[修改代码使用]codex官方app中使用中转(不需要cc-switch) 1.config.toml 2.sk方式登录
java