0

0

Java多线程安全共享变量与周期性输出实践指南

聖光之護

聖光之護

发布时间:2025-11-23 14:24:17

|

962人浏览过

|

来源于php中文网

原创

Java多线程安全共享变量与周期性输出实践指南

本教程深入探讨java多线程环境下,一个线程递增变量,另一个线程周期性打印的实现方法。文章阐述了共享变量的挑战及java内存模型,并提供了两种线程安全方案:利用atomicinteger进行原子操作以确保数据一致性,以及通过linkedblockingqueue实现生产者-消费者模式进行线程间通信,从而有效解决并发问题。

在多线程编程中,实现一个线程递增共享变量,而另一个线程周期性地读取并打印该变量的值,是一个常见的需求。然而,直接在多个线程间共享一个基本类型变量(如int)并进行读写操作,往往会遇到线程安全问题,导致数据不一致或不可预测的结果。本教程将深入探讨这一挑战,并提供两种健壮且常用的解决方案。

理解并发共享变量的挑战

当多个线程同时访问和修改同一个变量时,如果不采取适当的同步机制,就会出现问题。这主要是因为Java虚拟机(JVM)为了优化性能,可能会对指令进行重排序,或者允许每个线程拥有变量的本地缓存副本。这意味着一个线程对变量的修改,可能不会立即对其他线程可见。

考虑以下一个简单的、不安全的共享变量示例:

class UnsafeCounter {
    int counter = 0;

    public void increment() {
        counter++;
    }

    public int getCounter() {
        return counter;
    }
}

public class UnsafeSharedVariableDemo {
    public static void main(String[] args) throws InterruptedException {
        UnsafeCounter unsafeCounter = new UnsafeCounter();

        // 线程1:递增计数器
        Thread incrementer = new Thread(() -> {
            for (int i = 0; i < 10000; i++) {
                unsafeCounter.increment();
            }
            System.out.println("Incrementer finished. Final count (from incrementer's perspective): " + unsafeCounter.getCounter());
        });

        // 线程2:周期性打印计数器
        Thread printer = new Thread(() -> {
            try {
                for (int i = 0; i < 10; i++) {
                    System.out.println("Printer read counter: " + unsafeCounter.getCounter());
                    Thread.sleep(100); // 每100毫秒打印一次
                }
            } catch (InterruptedException e) {
                Thread.currentThread().interrupt();
            }
        });

        incrementer.start();
        printer.start();

        incrementer.join(); // 等待递增线程结束
        printer.join();     // 等待打印线程结束
        System.out.println("Main thread: Final counter value: " + unsafeCounter.getCounter());
    }
}

在上述代码中,printer线程可能会多次打印出相同的旧值,甚至在incrementer线程已经修改了counter之后。这是因为Java内存模型(JMM)规定,没有同步机制时,一个线程对共享变量的修改不保证对其他线程立即可见。为了解决这个问题,我们需要建立“Happens-Before”关系,确保操作的可见性和有序性。

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

方案一:使用原子类进行线程安全操作

java.util.concurrent.atomic包中的原子类提供了无锁的线程安全操作。它们通过使用底层的硬件指令(如CAS,Compare-And-Swap)来保证操作的原子性,并确保变量的修改对所有线程立即可见,从而建立Happens-Before关系。AtomicInteger是处理整数计数的理想选择。

以下是使用AtomicInteger实现线程安全计数器和周期性打印的示例:

import java.util.concurrent.atomic.AtomicInteger;

public class AtomicCounterDemo {
    private final AtomicInteger counter = new AtomicInteger(0); // 使用AtomicInteger作为共享计数器

    public void startThreads() {
        // 线程1:递增计数器
        Thread incrementerThread = new Thread(() -> {
            try {
                System.out.println("Incrementer thread started.");
                for (int i = 0; i < 50; i++) {
                    counter.incrementAndGet(); // 原子性递增
                    Thread.sleep(50); // 模拟工作负载
                }
                System.out.println("Incrementer thread finished. Final count: " + counter.get());
            } catch (InterruptedException e) {
                Thread.currentThread().interrupt();
                System.err.println("Incrementer thread interrupted.");
            }
        }, "Incrementer");

        // 线程2:周期性打印计数器
        Thread printerThread = new Thread(() -> {
            try {
                System.out.println("Printer thread started.");
                while (!Thread.currentThread().isInterrupted()) {
                    System.out.println("Current counter value: " + counter.get()); // 原子性获取
                    Thread.sleep(200); // 每200毫秒打印一次
                }
            } catch (InterruptedException e) {
                System.out.println("Printer thread interrupted. Exiting loop.");
            } finally {
                System.out.println("Printer thread finished.");
            }
        }, "Printer");

        incrementerThread.start();
        printerThread.start();

        // 等待递增线程完成
        try {
            incrementerThread.join();
        } catch (InterruptedException e) {
            Thread.currentThread().interrupt();
        }

        // 递增线程完成后,中断打印线程
        printerThread.interrupt();
        try {
            printerThread.join();
        } catch (InterruptedException e) {
            Thread.currentThread().interrupt();
        }

        System.out.println("All threads finished. Final counter value from main: " + counter.get());
    }

    public static void main(String[] args) {
        new AtomicCounterDemo().startThreads();
    }
}

工作原理与优点:

笔尖Ai写作
笔尖Ai写作

AI智能写作,1000+写作模板,轻松原创,拒绝写作焦虑!一款在线Ai写作生成器

下载
  • AtomicInteger内部通过CAS操作确保了incrementAndGet()和get()等方法的原子性。
  • 每次对AtomicInteger的修改都会刷新到主内存,并且读取操作会从主内存获取最新值,从而保证了线程间的可见性。
  • 相较于使用synchronized关键字进行方法或代码块锁定,原子类通常在竞争不激烈时性能更优,因为它避免了操作系统级别的线程上下文切换开销。

方案二:通过消息队列实现线程间通信

除了直接共享变量,另一种更解耦、更健壮的线程间通信方式是使用消息队列。一个线程作为“生产者”将数据放入队列,另一个线程作为“消费者”从队列中取出数据。java.util.concurrent包提供了多种并发集合,其中LinkedBlockingQueue是一个常用的选择,它支持阻塞式的存取操作,非常适合实现生产者-消费者模式。

以下是使用LinkedBlockingQueue实现计数器和周期性打印的示例:

import java.util.concurrent.BlockingQueue;
import java.util.concurrent.LinkedBlockingQueue;

public class MessageQueueCounterDemo {
    private final BlockingQueue queue = new LinkedBlockingQueue<>(10); // 容量为10的阻塞队列

    public void startThreads() {
        // 线程1:生产者 - 递增并放入队列
        Thread producerThread = new Thread(() -> {
            try {
                System.out.println("Producer thread started.");
                for (int i = 1; i <= 50; i++) {
                    queue.put(i); // 将当前计数值放入队列,如果队列满则阻塞
                    System.out.println("Producer added: " + i + ", Queue size: " + queue.size());
                    Thread.sleep(50); // 模拟生产数据的耗时
                }
                System.out.println("Producer finished adding 50 items. Signaling consumer to stop.");
                queue.put(-1); // 发送一个特殊值作为结束信号
            } catch (InterruptedException e) {
                Thread.currentThread().interrupt();
                System.err.println("Producer thread interrupted.");
            } finally {
                System.out.println("Producer thread finished.");
            }
        }, "Producer");

        // 线程2:消费者 - 周期性从队列取出并打印
        Thread consumerThread = new Thread(() -> {
            try {
                System.out.println("Consumer thread started.");
                while (true) {
                    Integer value = queue.take(); // 从队列取出数据,如果队列空则阻塞
                    if (value == -1) { // 收到结束信号
                        System.out.println("Consumer received termination signal. Exiting.");
                        break;
                    }
                    System.out.println("Consumer retrieved: " + value + ", Queue size: " + queue.size());
                    Thread.sleep(200); // 模拟消费数据的耗时
                }
            } catch (InterruptedException e) {
                System.out.println("Consumer thread interrupted. Exiting loop.");
            } finally {
                System.out.println("Consumer thread finished.");
            }
        }, "Consumer");

        producerThread.start();
        consumerThread.start();

        // 等待生产者和消费者线程完成
        try {
            producerThread.join();
            consumerThread.join();
        } catch (InterruptedException e) {
            Thread.currentThread().interrupt();
        }

        System.out.println("All threads finished. Final queue size: " + queue.size());
    }

    public static void main(String[] args) {
        new MessageQueueCounterDemo().startThreads();
    }
}

工作原理与优点:

  • LinkedBlockingQueue是线程安全的,其put()和take()方法内部处理了所有的同步细节,确保了数据的一致性。
  • 当队列为空时,take()方法会阻塞消费者线程,直到有数据可用;当队列已满时,put()方法会阻塞生产者线程,直到有空间可用。这天然地实现了线程间的协调。
  • 这种模式解耦了生产者和消费者,它们只需要知道队列的存在,而无需直接访问对方的状态。这使得系统更具弹性、可扩展性更强。
  • 适用于复杂的生产者-消费者场景,例如数据处理管道、任务调度等。

总结与注意事项

在Java中实现多线程共享变量并周期性输出,关键在于正确处理线程安全和数据可见性。

  1. 原子类(AtomicInteger等)

    • 优点:简单直接,适用于单一变量的原子操作,通常比synchronized有更好的性能(在低竞争环境下)。
    • 适用场景:计数器、标志位、简单状态的原子更新。
    • 注意事项:原子类只能保证单个操作的原子性。如果需要对多个变量进行复合操作,仍需其他同步机制(如synchronized或Lock)。
  2. 消息队列(LinkedBlockingQueue等)

    • 优点:高度解耦,天然支持生产者-消费者模式,易于扩展和维护,处理复杂数据流和任务调度非常有效。
    • 适用场景:任务队列、事件驱动系统、需要缓冲和流量控制的场景。
    • 注意事项:引入了额外的对象(队列元素),可能会有轻微的性能开销。需要设计好消息协议(例如本例中的-1作为结束信号)。

在实际开发中,应根据具体的业务需求、数据特性和性能要求来选择合适的并发工具。java.util.concurrent包是Java并发编程的核心,其中包含了大量强大的工具,如Semaphore、CountDownLatch、CyclicBarrier、ExecutorService等,深入学习这些工具将极大地提升多线程编程的能力。始终记住,并发编程是复杂的,彻底理解Java内存模型和Happens-Before关系是编写正确、高效并发代码的基础。

相关专题

更多
java
java

Java是一个通用术语,用于表示Java软件及其组件,包括“Java运行时环境 (JRE)”、“Java虚拟机 (JVM)”以及“插件”。php中文网还为大家带了Java相关下载资源、相关课程以及相关文章等内容,供大家免费下载使用。

842

2023.06.15

java正则表达式语法
java正则表达式语法

java正则表达式语法是一种模式匹配工具,它非常有用,可以在处理文本和字符串时快速地查找、替换、验证和提取特定的模式和数据。本专题提供java正则表达式语法的相关文章、下载和专题,供大家免费下载体验。

742

2023.07.05

java自学难吗
java自学难吗

Java自学并不难。Java语言相对于其他一些编程语言而言,有着较为简洁和易读的语法,本专题为大家提供java自学难吗相关的文章,大家可以免费体验。

740

2023.07.31

java配置jdk环境变量
java配置jdk环境变量

Java是一种广泛使用的高级编程语言,用于开发各种类型的应用程序。为了能够在计算机上正确运行和编译Java代码,需要正确配置Java Development Kit(JDK)环境变量。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

397

2023.08.01

java保留两位小数
java保留两位小数

Java是一种广泛应用于编程领域的高级编程语言。在Java中,保留两位小数是指在进行数值计算或输出时,限制小数部分只有两位有效数字,并将多余的位数进行四舍五入或截取。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

400

2023.08.02

java基本数据类型
java基本数据类型

java基本数据类型有:1、byte;2、short;3、int;4、long;5、float;6、double;7、char;8、boolean。本专题为大家提供java基本数据类型的相关的文章、下载、课程内容,供大家免费下载体验。

446

2023.08.02

java有什么用
java有什么用

java可以开发应用程序、移动应用、Web应用、企业级应用、嵌入式系统等方面。本专题为大家提供java有什么用的相关的文章、下载、课程内容,供大家免费下载体验。

431

2023.08.02

java在线网站
java在线网站

Java在线网站是指提供Java编程学习、实践和交流平台的网络服务。近年来,随着Java语言在软件开发领域的广泛应用,越来越多的人对Java编程感兴趣,并希望能够通过在线网站来学习和提高自己的Java编程技能。php中文网给大家带来了相关的视频、教程以及文章,欢迎大家前来学习阅读和下载。

16926

2023.08.03

Golang 性能分析与pprof调优实战
Golang 性能分析与pprof调优实战

本专题系统讲解 Golang 应用的性能分析与调优方法,重点覆盖 pprof 的使用方式,包括 CPU、内存、阻塞与 goroutine 分析,火焰图解读,常见性能瓶颈定位思路,以及在真实项目中进行针对性优化的实践技巧。通过案例讲解,帮助开发者掌握 用数据驱动的方式持续提升 Go 程序性能与稳定性。

6

2026.01.22

热门下载

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

精品课程

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

共23课时 | 2.8万人学习

C# 教程
C# 教程

共94课时 | 7.3万人学习

Java 教程
Java 教程

共578课时 | 49.2万人学习

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

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