0

0

Java多线程并发控制:使用synchronized关键字实现互斥访问

碧海醫心

碧海醫心

发布时间:2025-10-07 09:34:21

|

839人浏览过

|

来源于php中文网

原创

Java多线程并发控制:使用synchronized关键字实现互斥访问

本文旨在解决Java多线程环境下因并发执行导致的操作中断问题,特别是当多个线程尝试同时访问共享资源(如打印输出)时。我们将深入探讨如何通过Java的synchronized关键字和共享锁对象来确保代码段的互斥执行,从而避免中断和数据不一致,并解释为何线程优先级并非解决此类问题的理想方案。

1. 理解多线程并发中的挑战

在多线程编程中,当多个线程同时访问或修改共享资源时,可能会出现竞态条件(race condition),导致数据不一致或操作中断。例如,一个线程正在执行一个多步操作(如打印一条完整的日志信息),而另一个线程突然插入并执行自己的操作,就会导致输出混乱或不完整。开发者有时会尝试通过设置线程优先级来解决这类问题,期望高优先级的线程能够优先完成其任务而不被中断。然而,线程优先级在java中通常不可靠,其行为高度依赖于底层操作系统和jvm实现,并不能保证严格的执行顺序或互斥访问。

2. 线程优先级的局限性

Java提供了Thread.setPriority()方法来设置线程的优先级,范围从Thread.MIN_PRIORITY到Thread.MAX_PRIORITY。理论上,高优先级的线程会比低优先级的线程获得更多的CPU时间片。然而,在实际应用中,线程优先级存在以下局限:

  • 平台依赖性: 线程优先级在不同操作系统上的实现差异很大。某些操作系统可能完全忽略Java设置的优先级,或者只支持有限的优先级级别映射。
  • 非确定性: 即使在支持优先级的系统上,也不能保证高优先级线程总是先于低优先级线程执行,或者不会被中断。它更多的是一种调度提示,而非严格的执行保证。
  • 无法解决互斥问题: 优先级无法阻止两个线程同时进入临界区,从而无法解决竞态条件问题。它仅仅影响线程获取CPU时间的可能性,而不是对共享资源的访问控制。

因此,对于需要确保操作原子性或互斥访问共享资源的场景,线程优先级并非合适的解决方案。

3. 解决方案:使用synchronized关键字实现互斥访问

为了确保在多线程环境下对共享资源的访问是互斥的,Java提供了synchronized关键字。synchronized可以用于方法或代码块,它通过使用一个内部锁机制来保证在同一时间只有一个线程可以执行被synchronized保护的代码。

当一个线程进入一个synchronized代码块或方法时,它会尝试获取与该代码块或方法关联的锁。如果锁已被其他线程持有,当前线程就会被阻塞,直到锁被释放。一旦当前线程获取到锁,它就可以独占地执行synchronized代码,直到执行完毕并释放锁。

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

3.1 synchronized代码块的用法

对于需要保护特定代码段免受并发访问的场景,通常使用synchronized代码块。它需要一个对象作为锁。

示例代码:

public class PrinterManager {

    // 定义一个共享的锁对象,通常声明为final,避免被重新赋值
    public static final Object MY_LOCK = new Object();

    public void printMessage1(String message) {
        // 使用MY_LOCK对象作为锁,确保printMessage1方法中的打印操作是互斥的
        synchronized (MY_LOCK) {
            System.out.print("[线程 " + Thread.currentThread().getName() + "] 开始打印: ");
            try {
                // 模拟打印过程中的耗时操作
                Thread.sleep(50);
            } catch (InterruptedException e) {
                Thread.currentThread().interrupt();
            }
            System.out.println(message + " (结束)");
        }
    }

    public void printMessage2(String message) {
        // 同样使用MY_LOCK对象作为锁,确保与printMessage1互斥
        synchronized (MY_LOCK) {
            System.out.print("[线程 " + Thread.currentThread().getName() + "] 开始打印: ");
            try {
                Thread.sleep(50);
            } catch (InterruptedException e) {
                Thread.currentThread().interrupt();
            }
            System.out.println(message + " (结束)");
        }
    }

    public static void main(String[] args) {
        PrinterManager manager = new PrinterManager();

        // 创建并启动多个线程,它们将尝试同时调用printMessage方法
        for (int i = 0; i < 5; i++) {
            final int taskId = i;
            new Thread(() -> {
                manager.printMessage1("任务A-" + taskId);
            }, "Thread-" + i + "-A").start();

            new Thread(() -> {
                manager.printMessage2("任务B-" + taskId);
            }, "Thread-" + i + "-B").start();
        }
    }
}

在上述示例中:

Manus
Manus

全球首款通用型AI Agent,可以将你的想法转化为行动。

下载
  • public static final Object MY_LOCK = new Object(); 定义了一个静态的、final的锁对象。静态确保所有PrinterManager实例共享同一个锁,final确保锁对象不会在运行时被替换。
  • synchronized (MY_LOCK) { ... } 块表示任何线程在执行这段代码之前,都必须先获取MY_LOCK对象的锁。
  • 由于printMessage1和printMessage2方法都使用了同一个MY_LOCK对象进行同步,因此在任何给定时间,只有一个线程能够执行这两个方法中被synchronized保护的代码段。这有效地防止了打印输出被其他线程中断,确保了每条消息的完整性。

3.2 synchronized方法的用法

当需要同步整个方法时,可以直接在方法声明上使用synchronized关键字。对于实例方法,锁是方法所属的实例对象(this);对于静态方法,锁是方法所属的类的Class对象。

public class SynchronizedMethodExample {

    // 实例方法,锁是SynchronizedMethodExample的实例对象
    public synchronized void instanceMethod() {
        System.out.println("实例方法 - 线程: " + Thread.currentThread().getName());
        // ... 临界区代码 ...
    }

    // 静态方法,锁是SynchronizedMethodExample.class对象
    public static synchronized void staticMethod() {
        System.out.println("静态方法 - 线程: " + Thread.currentThread().getName());
        // ... 临界区代码 ...
    }
}

注意: 当使用synchronized方法时,需要确保所有需要互斥访问的方法都使用相同的锁(即同一个实例对象或同一个Class对象)。

4. 注意事项与最佳实践

  • 选择合适的锁对象:

    • 对于实例方法或访问实例变量的临界区,通常使用this作为锁,或定义一个private final Object lock = new Object();作为成员变量锁。
    • 对于静态方法或访问静态变量的临界区,必须使用Class对象作为锁(例如MyClass.class),或定义一个private static final Object STATIC_LOCK = new Object();作为静态锁。
    • 避免使用可变对象作为锁,因为锁对象一旦改变,同步机制就会失效。
    • 避免使用String字面量作为锁,因为字符串常量池可能导致意外的死锁或同步问题。
  • 锁的粒度: 锁的粒度应尽可能小,只保护真正需要同步的代码块。过大的锁粒度会降低并发性,因为线程会花费更多时间等待锁的释放。

  • 避免死锁: 当两个或多个线程互相持有对方所需的锁时,就会发生死锁。设计同步机制时应仔细考虑锁的获取顺序,尽量保持一致。

  • 性能考量: synchronized是Java提供的开箱即用的同步机制,使用方便。但在高并发场景下,其性能可能不如java.util.concurrent.locks.ReentrantLock等显式锁。ReentrantLock提供了更细粒度的控制,如公平锁、条件变量等,但使用起来也更复杂。

5. 总结

在Java多线程编程中,解决因并发访问共享资源导致的操作中断和数据不一致问题,应优先采用synchronized关键字或java.util.concurrent包下的锁机制。线程优先级并非用于实现严格的互斥或控制执行顺序的可靠方法。通过正确使用synchronized代码块和共享锁对象,我们可以有效地保护临界区,确保关键操作的原子性,从而构建健壮、可靠的多线程应用程序。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

463

2023.08.02

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1502

2023.10.24

字符串常量的表示方法
字符串常量的表示方法

字符串常量的表示方法:1、使用引号;2、转义字符;3、多行字符串;4、原始字符串;5、字符串连接;6、字符串字面量和对象;7、编码问题。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

140

2023.12.26

js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

298

2023.08.03

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

212

2023.09.04

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1502

2023.10.24

字符串介绍
字符串介绍

字符串是一种数据类型,它可以是任何文本,包括字母、数字、符号等。字符串可以由不同的字符组成,例如空格、标点符号、数字等。在编程中,字符串通常用引号括起来,如单引号、双引号或反引号。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

624

2023.11.24

java读取文件转成字符串的方法
java读取文件转成字符串的方法

Java8引入了新的文件I/O API,使用java.nio.file.Files类读取文件内容更加方便。对于较旧版本的Java,可以使用java.io.FileReader和java.io.BufferedReader来读取文件。在这些方法中,你需要将文件路径替换为你的实际文件路径,并且可能需要处理可能的IOException异常。想了解更多java的相关内容,可以阅读本专题下面的文章。

633

2024.03.22

java入门学习合集
java入门学习合集

本专题整合了java入门学习指南、初学者项目实战、入门到精通等等内容,阅读专题下面的文章了解更多详细学习方法。

1

2026.01.29

热门下载

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

精品课程

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

共23课时 | 3万人学习

C# 教程
C# 教程

共94课时 | 7.9万人学习

Java 教程
Java 教程

共578课时 | 53.2万人学习

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

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