【JAVA学习之路 | 提高篇】线程的通信

简介: 【JAVA学习之路 | 提高篇】线程的通信

1.需求(为什么需要线程通信)

当我们需要多个线程完成同一任务时,并且希望他们有规律的执行,那么多线程之间需要一些通信机制,并且可以协调他们的工作,以此实现多个线程共同操作共享数据.

例 : A做包子,B吃包子,包子相当于共享操作的数据,B必须等到A做好才能吃,那么线程AB间就需要通信.即等待唤醒机制.

2.等待唤醒机制

这是多线程的一种协同的机制.谈到线程我们常常想到是多个线程竞争一把锁.但不完全是这样.线程之间也可以实现协同.


在一个线程满足某个条件时,就进入等待状态(wait),等待其他线程进行到某处指定代码时,将等待状态的线程唤醒(notify).可以指定wait的时间,过了这个时间可以自动唤醒@也可以使用notifyAll来唤醒所有等待的线程.

  • wait : 线程不在参与活动,将释放CPU调度.进入wait set中,不会浪费CPU资源.也不会去竞争锁.此时线程状态为WAITING或者是TIME_WAITING.他等待其他线程执行某个操作(notify/wait(time)执行了time的时间),才能够将其从wait set中释放出来,重新参与CPU调度.
  • notify : 选取待在wait set中的线程将其释放重新参与CPU调度.
  • notifyAll : 将全部待在wait set中的线程全部释放参与CPU调度.

3.例 : 交替打印出输出的一百以内的数.

解释 : 调用start方法启动线程.假设是线程1(当然也可能是线程2)进入while语句并握住锁,进行输出语句后,调用wait()释放锁,线程2握住锁,进行到notify()唤醒线程1重新参与CPU调度......

public class WaitNotifyTest {
    public static void main(String[] args) {
        WaitNotify w = new WaitNotify();
        Thread t1 = new Thread(w, "线程-1");
        Thread t2 = new Thread(w, "线程-2");
        t1.start();
        t2.start();
    }
}
class WaitNotify implements Runnable {
    int change = 100;
    Object obj = new Object();
    @Override
    public void run() {
        while (true) {
            synchronized(obj) {
                obj.notify();
                try {
                    Thread.sleep(100);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                if (change > 0) {
                    System.out.println(Thread.currentThread().getName() + "\t" + change);
                    change--;
                }
                try{
                    obj.wait();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
 
        }
    }
}
 
控制台
线程-1  100
线程-2  99
线程-1  98
线程-2  97
线程-1  96
线程-2  95
线程-1  94

注 :

  • 这三个方法的使用必须都要在同步代码块或者同步方法中.
  • 此三个方法的调用者必须是同步监视器,否则会触发异常.
  • 按住Ctrl键,可以查看该三个方法的源码源码.


public final native void notify();
public final void wait() throws InterruptedException {
    wait(0L);
}

此三个方法都声明在Object类中,Ctrl+F12,查看Object类.

  • 因为同步监视器可以是任何对象(保证线程安全性需要满足唯一),由于该三个方法是Object类中的方法,所以同步监视器可以调用.

4.wait()与sleep()的区别.

(1). 查看源码 :

public static native void sleep(long millis) throws InterruptedException;
public final void wait() throws InterruptedException {
        wait(0L);
    }

(2) 相同点 :

一旦执行,都可以阻塞线程.

(3) 不同点 :

  • 声明的位置不同 : sleep()方法声明在Thread类中,而且是静态的方法.wait()方法声明在Object类中.
  • 使用的场景不同 : sleep()可以声明在任何所需要的场景.wait()必须声明在同步代码块或同步方法中(因为该写方法必须通过同步监视器调用).
  • CPU调度 : 使用sleep()并不会释放CPU调度,只是线程阻塞.而wait()阻塞线程,并且释放该线程的CPU调度.
  • 结束阻塞方式 : sleep(time)方法在时间time之后结束阻塞.wait(time)方法在时间time之后或者notify/notifyAll结束阻塞.
相关文章
|
7天前
|
监控 安全 Java
在 Java 中使用线程池监控以及动态调整线程池时需要注意什么?
【10月更文挑战第22天】在进行线程池的监控和动态调整时,要综合考虑多方面的因素,谨慎操作,以确保线程池能够高效、稳定地运行,满足业务的需求。
77 38
|
4天前
|
安全 Java
java 中 i++ 到底是否线程安全?
本文通过实例探讨了 `i++` 在多线程环境下的线程安全性问题。首先,使用 100 个线程分别执行 10000 次 `i++` 操作,发现最终结果小于预期的 1000000,证明 `i++` 是线程不安全的。接着,介绍了两种解决方法:使用 `synchronized` 关键字加锁和使用 `AtomicInteger` 类。其中,`AtomicInteger` 通过 `CAS` 操作实现了高效的线程安全。最后,通过分析字节码和源码,解释了 `i++` 为何线程不安全以及 `AtomicInteger` 如何保证线程安全。
java 中 i++ 到底是否线程安全?
|
6天前
|
安全 Java
在 Java 中使用实现 Runnable 接口的方式创建线程
【10月更文挑战第22天】通过以上内容的介绍,相信你已经对在 Java 中如何使用实现 Runnable 接口的方式创建线程有了更深入的了解。在实际应用中,需要根据具体的需求和场景,合理选择线程创建方式,并注意线程安全、同步、通信等相关问题,以确保程序的正确性和稳定性。
|
5天前
|
缓存 Java 调度
Java中的多线程编程:从基础到实践
【10月更文挑战第24天】 本文旨在为读者提供一个关于Java多线程编程的全面指南。我们将从多线程的基本概念开始,逐步深入到Java中实现多线程的方法,包括继承Thread类、实现Runnable接口以及使用Executor框架。此外,我们还将探讨多线程编程中的常见问题和最佳实践,帮助读者在实际项目中更好地应用多线程技术。
11 3
|
6天前
|
监控 安全 Java
Java多线程编程的艺术与实践
【10月更文挑战第22天】 在现代软件开发中,多线程编程是一项不可或缺的技能。本文将深入探讨Java多线程编程的核心概念、常见问题以及最佳实践,帮助开发者掌握这一强大的工具。我们将从基础概念入手,逐步深入到高级主题,包括线程的创建与管理、同步机制、线程池的使用等。通过实际案例分析,本文旨在提供一种系统化的学习方法,使读者能够在实际项目中灵活运用多线程技术。
|
5天前
|
缓存 安全 Java
Java中的多线程编程:从基础到实践
【10月更文挑战第24天】 本文将深入探讨Java中的多线程编程,包括其基本原理、实现方式以及常见问题。我们将从简单的线程创建开始,逐步深入了解线程的生命周期、同步机制、并发工具类等高级主题。通过实际案例和代码示例,帮助读者掌握多线程编程的核心概念和技术,提高程序的性能和可靠性。
8 2
|
5天前
|
Java
Java中的多线程编程:从基础到实践
本文深入探讨Java多线程编程,首先介绍多线程的基本概念和重要性,接着详细讲解如何在Java中创建和管理线程,最后通过实例演示多线程的实际应用。文章旨在帮助读者理解多线程的核心原理,掌握基本的多线程操作,并能够在实际项目中灵活运用多线程技术。
|
5天前
|
JavaScript Java 项目管理
Java毕设学习 基于SpringBoot + Vue 的医院管理系统 持续给大家寻找Java毕设学习项目(附源码)
基于SpringBoot + Vue的医院管理系统,涵盖医院、患者、挂号、药物、检查、病床、排班管理和数据分析等功能。开发工具为IDEA和HBuilder X,环境需配置jdk8、Node.js14、MySQL8。文末提供源码下载链接。
|
7天前
|
Java 数据处理 开发者
Java多线程编程的艺术:从入门到精通####
【10月更文挑战第21天】 本文将深入探讨Java多线程编程的核心概念,通过生动实例和实用技巧,引导读者从基础认知迈向高效并发编程的殿堂。我们将一起揭开线程管理的神秘面纱,掌握同步机制的精髓,并学习如何在实际项目中灵活运用这些知识,以提升应用性能与响应速度。 ####
28 3
|
6天前
|
Java 开发者
Java中的多线程基础与应用
【10月更文挑战第24天】在Java的世界中,多线程是提高效率和实现并发处理的关键。本文将深入浅出地介绍如何在Java中创建和管理多线程,以及如何通过同步机制确保数据的安全性。我们将一起探索线程生命周期的奥秘,并通过实例学习如何优化多线程的性能。无论你是初学者还是有一定经验的开发者,这篇文章都将为你打开一扇通往高效编程的大门。
10 0