墨风如雪博客

  • 源码小店
  • 导航站
  • 登录
  • java
  • 资源分享
让AI使用变得如此简单
  1. 首页
  2. java
  3. 正文

Java中synchronized关键字的八个锁问题及解决办法

2023年 5月 2日 199点热度 0人点赞 0条评论

八个锁问题

当多个线程访问同一个对象的synchronized方法时,可能会产生八种不同的锁问题:

  1. 对于普通同步方法,锁是当前实例对象。
  2. 对于静态同步方法,锁是当前类的Class对象。
  3. 对于同步方法块,锁是synchronized括号里配置的对象。
  4. 对于一个普通方法,即非同步方法,不会产生锁问题,不需要等待锁,直接访问方法即可。
  5. 对于一个静态方法,即使非同步方法,也会存在锁问题。因为静态方法是属于类的,而不是属于实例的,所以静态方法的锁是当前类的Class对象。
  6. 当一个线程执行的代码出现异常时,锁会被自动释放。
  7. synchronized锁住的代码块不是原子操作。如果同一个线程执行两次synchronized代码块,它将获得锁两次。
  8. 如果两个及以上线程同时等待某个monitor对象,那么只有一个线程可以继续执行,其他线程将被阻塞。

解决办法

下面分别对八个锁问题给出相应的解决办法。

1. 对于普通同步方法,锁是当前实例对象。

解决办法是在方法前面加上synchronized关键字。

public synchronized void add(){...}

2. 对于静态同步方法,锁是当前类的Class对象。

解决办法是在方法前面加上synchronized关键字,同时锁定当前类。

public static synchronized void add(){...}

3. 对于同步方法块,锁是synchronized括号里配置的对象。

解决办法是在同步代码块前面加上synchronized关键字,同时锁定指定的对象。例如:

public void add(){
    synchronized(lock){
        ...
    }
}

4. 对于普通非同步方法,不会产生锁问题,无需解决。

直接访问即可。

5. 对于静态非同步方法,也会存在锁问题。

解决办法是锁定当前类的Class对象,可以使用synchronized(Class)。

public static void add(){
    synchronized(ClassName.class){
        ...
    }
}

6. 当一个线程执行的代码出现异常时,锁会被自动释放。

无需解决,JVM会自动释放锁。

7. synchronized锁住的代码块不是原子操作。

解决办法是使用同步方法而不是同步代码块。

public synchronized void add(){
    ...
}

8. 如果两个及以上线程同时等待某个monitor对象,那么只有一个线程可以继续执行,其他线程将被阻塞。

解决办法是通过使用notify()或者notifyAll()通知等待线程,从而释放对象锁。

public synchronized void add(){
    ...
    notify();
}

下面是一个使用synchronized解决八个锁问题的示例代码:

public class MyObject {
    private static int count1 = 0;
    private int count2 = 0;
    private Object lock = new Object();

    public static synchronized void method1() {
        // 锁定当前类,等价于 synchronized(MyObject.class)
        System.out.println("Method 1 start");
        try {
            Thread.sleep(3000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println("Method 1 end");
    }

    public synchronized void method2() {
        // 锁定当前实例对象,等价于 synchronized(this)
        System.out.println("Method 2 start");
        try {
            Thread.sleep(3000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println("Method 2 end");
    }

    public void method3() {
        // 锁定指定对象
        synchronized (lock) {
            System.out.println("Method 3 start");
            try {
                Thread.sleep(3000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println("Method 3 end");
        }
    }

    public static void method4() {
        // 锁定当前类,等价于 synchronized(MyObject.class)
        synchronized (MyObject.class) {
            System.out.println("Method 4 start");
            try {
                Thread.sleep(3000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println("Method 4 end");
        }
    }

    public static synchronized void method5() {
        // 锁定当前类,等价于 synchronized(MyObject.class)
        count1++;
        System.out.println("Method 5 count1=" + count1);
        try {
            Thread.sleep(3000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }

    public synchronized void method6() {
        // 锁定当前实例对象,等价于 synchronized(this)
        count2++;
        System.out.println("Method 6 count2=" + count2);
        try {
            Thread.sleep(3000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }

    public static void method7() {
        // 锁定当前类,等价于 synchronized(MyObject.class)
        synchronized (MyObject.class) {
            System.out.println("Method 7 start");
            try {
                Thread.sleep(3000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println("Method 7 end");
        }
    }

    public static synchronized void method8() {
        // 锁定当前类,等价于 synchronized(MyObject.class)
        System.out.println("Method 8 start");
        try {
            Thread.sleep(3000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println("Method 8 end");
    }
}
本作品采用 知识共享署名 4.0 国际许可协议 进行许可
标签: java synchronized 八个锁问题 解决办法
最后更新:2023年 5月 2日

墨风如雪

一个热爱生活,热爱分享的程序员

打赏 点赞
< 上一篇
下一篇 >

文章评论

您需要 登录 之后才可以评论

墨风如雪

一个热爱生活,热爱分享的程序员

最新 热点 随机
最新 热点 随机
AI“游侠”降临A股:16个“大脑”组团“炒股”,30秒“算命”市场! 视频魔法来了!AI能实时“变脸”直播,连游戏画面也能瞬间换装? 告别“听指令”,AI要“自己动手”了!ChatGPT Agent,AI界的“全能选手”已上线! 8B 模型吊打 671B?数学证明界“卷王”Goedel-Prover-V2 来了! Kiro来了!亚马逊放大招,软件开发要被AI“绑架”了吗? 火速围观!Trae IDE 迎来两大明星模型,Kimi K2 硬核登场,Grok-4 (Beta) 闪耀国际!
昆仑万维扔出王炸:32B模型干翻671B,代码界迎来全能修理工!8亿参数撬动实时混音!谷歌开源“口袋DJ”,人人都能玩转音乐告别插件时代!OmniGen2:一个模型,通吃所有AIGC神操作2000万次呼唤背后,蓝骑士有了“赛博外挂”智能触手可及:Google Gemma-3n 系列模型,让万物皆能“思考”AI圈大地震!120亿参数的FLUX编辑器开源,你的显卡准备好了吗?
告别“听指令”,AI要“自己动手”了!ChatGPT Agent,AI界的“全能选手”已上线! Google 暂时停止 Gemini 2.5 Pro 免费 API 访问 告诉你spring boot 的生命周期是怎么样的(超详细) docker-compose使用详解 SpringMVC | SpringMVC 入门 java 消息队列框架Apache Kafka的(超详细总结)
标签聚合
spring 大模型 教程 AI deepseek 设计模式 算法 java

COPYRIGHT © 2023 墨风如雪博客. ALL RIGHTS RESERVED.

Theme Kratos Made By Seaton Jiang

免责声明 - 隐私政策