0

0

Java中synchronized关键字怎么用 详解Java同步锁的4种使用方法

穿越時空

穿越時空

发布时间:2025-06-19 08:21:01

|

974人浏览过

|

来源于php中文网

原创

synchronized关键字在java中用于实现线程同步,确保多线程并发访问共享资源时的互斥执行。其主要使用方式包括:1. 同步代码块,通过指定对象作为锁;2. 同步方法,锁为当前对象(this)或类对象(class);3. 静态同步方法,等价于使用类对象作为锁;4. 同步静态变量,通常使用静态对象作为锁。此外,synchronized依赖jvm的monitor机制,通过monitorenter和monitorexit指令实现锁的获取与释放,并在jdk 1.6后通过锁优化提升了性能。相比reentrantlock,synchronized是隐式锁,自动释放,而reentrantlock需要手动释放且提供更多功能,但更易出错。避免死锁的方法包括统一锁顺序、限制锁持有时间、使用定时锁及工具检测。

Java中synchronized关键字怎么用 详解Java同步锁的4种使用方法

synchronized关键字在Java中用于实现线程同步,确保多个线程在并发访问共享资源时,只有一个线程可以执行特定代码块或方法,从而避免数据竞争和不一致性。它本质上提供了一种互斥锁机制。

Java中synchronized关键字怎么用 详解Java同步锁的4种使用方法

解决方案

Java中synchronized关键字怎么用 详解Java同步锁的4种使用方法

synchronized关键字主要有四种使用方式:

立即学习Java免费学习笔记(深入)”;

Java中synchronized关键字怎么用 详解Java同步锁的4种使用方法
  1. 同步代码块(Synchronized Block):

    通过synchronized(object)来指定一个对象作为锁。任何时刻,只能有一个线程获得该对象的锁,其他线程必须等待。

    public class Counter {
        private int count = 0;
        private Object lock = new Object(); // 锁对象
    
        public void increment() {
            synchronized (lock) {
                count++;
            }
        }
    
        public int getCount() {
            return count;
        }
    }

    这里,lock对象充当了锁,只有获得lock锁的线程才能执行count++操作。 选择哪个对象作为锁至关重要。 如果多个线程访问不同的Counter实例,那么每个实例都有自己的lock对象,同步就无效了。

  2. 同步方法(Synchronized Method):

    将整个方法声明为synchronized。 对于实例方法,锁是当前对象(this);对于静态方法,锁是当前类(Class对象)。

    public class Counter {
        private int count = 0;
    
        public synchronized void increment() {
            count++;
        }
    
        public int getCount() {
            return count;
        }
    }

    等价于:

    Autoenhance
    Autoenhance

    AI照片编辑器

    下载
    public class Counter {
        private int count = 0;
    
        public void increment() {
            synchronized (this) {
                count++;
            }
        }
    
        public int getCount() {
            return count;
        }
    }

    静态同步方法:

    public class Counter {
        private static int count = 0;
    
        public static synchronized void increment() {
            count++;
        }
    
        public static int getCount() {
            return count;
        }
    }

    等价于:

    public class Counter {
        private static int count = 0;
    
        public static void increment() {
            synchronized (Counter.class) {
                count++;
            }
        }
    
        public static int getCount() {
            return count;
        }
    }
  3. 同步静态变量(不常用):

    虽然不常见,但也可以通过同步静态变量来实现同步。 这本质上和同步代码块使用Class对象作为锁是一样的。

    public class Counter {
        private static int count = 0;
        private static Object staticLock = new Object();
    
        public static void increment() {
            synchronized (staticLock) {
                count++;
            }
        }
    
        public static int getCount() {
            return count;
        }
    }
  4. 隐式锁(Intrinsic Lock or Monitor Lock):

    当线程进入synchronized块或方法时,它会自动获得与对象关联的隐式锁。 当线程退出synchronized块或方法时,锁会被自动释放。 这种锁是可重入的,意味着同一个线程可以多次获得同一个锁而不会阻塞。

Synchronized 和 ReentrantLock 的区别是什么?

  • synchronized 是 Java 关键字,属于 JVM 层面,而 ReentrantLock 是一个类,属于 API 层面。
  • synchronized 会自动释放锁,而 ReentrantLock 需要手动释放锁(lock.unlock()),如果忘记释放,可能会导致死锁。
  • ReentrantLock 提供了更多的功能,例如可中断的锁、公平锁等,而 synchronized 相对简单。
  • 性能方面,在 JDK 1.6 之后,synchronized 进行了大量的优化,在低并发情况下,性能可能优于 ReentrantLock,但在高并发情况下,ReentrantLock 通常性能更好。 理论上,ReentrantLock更灵活,但用不好也更容易出错。

如何避免死锁?

死锁是指两个或多个线程互相持有对方需要的资源,导致所有线程都无法继续执行的情况。 避免死锁的一些常见策略:

  • 避免循环等待: 确保线程获取锁的顺序是一致的,避免形成循环依赖。
  • 限制锁的持有时间: 尽量缩短持有锁的时间,减少其他线程等待锁的机会。
  • 使用定时锁: 使用 ReentrantLocktryLock(long timeout, TimeUnit unit) 方法,设置超时时间,避免无限等待。
  • 死锁检测: 一些工具可以检测死锁,例如 JConsole 和 VisualVM。

举个例子,假设有两个线程 A 和 B,它们都需要获取锁 lock1 和 lock2。如果线程 A 先获取了 lock1,然后尝试获取 lock2,而线程 B 先获取了 lock2,然后尝试获取 lock1,那么就可能发生死锁。

Object lock1 = new Object();
Object lock2 = new Object();

// 线程 A
new Thread(() -> {
    synchronized (lock1) {
        System.out.println("线程 A 获取了 lock1");
        try { Thread.sleep(100); } catch (InterruptedException e) {}
        synchronized (lock2) {
            System.out.println("线程 A 获取了 lock2");
        }
    }
}).start();

// 线程 B
new Thread(() -> {
    synchronized (lock2) {
        System.out.println("线程 B 获取了 lock2");
        try { Thread.sleep(100); } catch (InterruptedException e) {}
        synchronized (lock1) {
            System.out.println("线程 B 获取了 lock1");
        }
    }
}).start();

在这个例子中,如果线程 A 获取了 lock1,线程 B 获取了 lock2,然后它们互相等待对方释放锁,就会发生死锁。 避免这种情况,可以确保两个线程以相同的顺序获取锁,例如都先获取 lock1,再获取 lock2。

Synchronized的底层实现原理是什么?

synchronized 的实现依赖于 JVM 的 monitor(监视器锁)机制。 每个 Java 对象都关联一个 monitor,当线程执行到 synchronized 块或方法时,会尝试获取 monitor 的所有权。 如果 monitor 没有被其他线程持有,则该线程成功获取 monitor,并将 monitor 的计数器加 1。 如果 monitor 已经被其他线程持有,则该线程会被阻塞,直到 monitor 的计数器变为 0,即持有 monitor 的线程释放了锁。

在 JVM 层面,synchronized 是通过 monitorentermonitorexit 指令来实现的。 monitorenter 指令用于获取 monitor 的所有权,monitorexit 指令用于释放 monitor 的所有权。 JVM 会确保 monitorentermonitorexit 指令是成对出现的,即使在发生异常的情况下,也会确保锁被释放,从而避免死锁。 早期的synchronized效率比较低,因为是重量级锁,会涉及到用户态和内核态的切换,开销较大。但是现在JVM对synchronized做了很多优化,比如锁升级,轻量级锁,偏向锁等,在很多场景下synchronized的效率已经很高了。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
counta和count的区别
counta和count的区别

Count函数用于计算指定范围内数字的个数,而CountA函数用于计算指定范围内非空单元格的个数。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

198

2023.11.20

class在c语言中的意思
class在c语言中的意思

在C语言中,"class" 是一个关键字,用于定义一个类。想了解更多class的相关内容,可以阅读本专题下面的文章。

469

2024.01.03

python中class的含义
python中class的含义

本专题整合了python中class的相关内容,阅读专题下面的文章了解更多详细内容。

13

2025.12.06

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

502

2023.08.10

Python 多线程与异步编程实战
Python 多线程与异步编程实战

本专题系统讲解 Python 多线程与异步编程的核心概念与实战技巧,包括 threading 模块基础、线程同步机制、GIL 原理、asyncio 异步任务管理、协程与事件循环、任务调度与异常处理。通过实战示例,帮助学习者掌握 如何构建高性能、多任务并发的 Python 应用。

166

2025.12.24

java多线程相关教程合集
java多线程相关教程合集

本专题整合了java多线程相关教程,阅读专题下面的文章了解更多详细内容。

10

2026.01.21

C++多线程相关合集
C++多线程相关合集

本专题整合了C++多线程相关教程,阅读专题下面的的文章了解更多详细内容。

14

2026.01.21

Python 自然语言处理(NLP)基础与实战
Python 自然语言处理(NLP)基础与实战

本专题系统讲解 Python 在自然语言处理(NLP)领域的基础方法与实战应用,涵盖文本预处理(分词、去停用词)、词性标注、命名实体识别、关键词提取、情感分析,以及常用 NLP 库(NLTK、spaCy)的核心用法。通过真实文本案例,帮助学习者掌握 使用 Python 进行文本分析与语言数据处理的完整流程,适用于内容分析、舆情监测与智能文本应用场景。

10

2026.01.27

拼多多赚钱的5种方法 拼多多赚钱的5种方法
拼多多赚钱的5种方法 拼多多赚钱的5种方法

在拼多多上赚钱主要可以通过无货源模式一件代发、精细化运营特色店铺、参与官方高流量活动、利用拼团机制社交裂变,以及成为多多进宝推广员这5种方法实现。核心策略在于通过低成本、高效率的供应链管理与营销,利用平台社交电商红利实现盈利。

109

2026.01.26

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 2.9万人学习

C# 教程
C# 教程

共94课时 | 7.8万人学习

Java 教程
Java 教程

共578课时 | 52.3万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号