深入了解线程:并发编程的核心要点和应用

什么是线程?

线程(Thread)是计算机科学中的一个基本概念,它是进程内的执行单元。线程是操作系统或进程内部的轻量级执行流,具有独立的程序计数器(PC)和栈,但共享相同进程的资源,包括内存空间、文件句柄等。线程是进程内的多个线程共享相同的进程上下文,因此它们可以更高效地协同工作。

Linux中使用 top -H -p <进程ID> 能得到这个进程下的线程
在这里插入图片描述

特点

1.轻量级: 与进程相比,线程通常更轻量级,因为它们共享大部分进程的资源,如内存空间。这使得线程的创建、销毁和切换成本较低。

2.并发执行: 多个线程可以在同一进程中并发执行。这意味着它们可以同时执行不同的任务,从而提高了程序的性能和响应时间。

3.共享资源: 线程在同一进程内共享相同的资源,包括内存、文件、打开的网络连接等。这可以简化资源管理和共享,但也需要进行适当的同步和互斥控制,以防止竞态条件和数据冲突。

4.独立执行流: 每个线程有自己的程序计数器(PC)和栈,可以独立执行不同的代码路径。这使得线程能够同时执行不同的函数或任务。

5.通信和同步: 在多线程应用程序中,线程之间需要进行通信和同步以协调它们的工作。常见的同步机制包括互斥锁、信号量、条件变量等。

6.并行性: 线程是实现并行计算的一种方式。多个线程可以同时执行不同的计算任务,从而加速程序的执行。

优点

1.并发性和并行性: 线程允许多个任务同时执行,从而提高程序的并发性和并行性。这有助于更有效地利用多核处理器和提高系统性能。在多核系统中,单线程程序无法充分利用所有可用的处理器核心,而多线程程序可以并行执行不同任务。

2.提高响应性: 多线程可以使应用程序更加响应用户输入或外部事件。例如,在图形用户界面(GUI)应用程序中,单线程可能会导致界面冻结,而多线程可以使界面保持响应,即使某个线程正在执行耗时的操作。

3.简化程序设计: 多线程可以使程序更模块化和易于理解。任务可以分解为多个线程,每个线程负责执行特定的工作。这有助于提高代码的可维护性和可扩展性。

4.资源共享: 线程在同一进程内共享相同的资源,包括内存、文件、网络连接等。这降低了资源分配的复杂性,减少了资源浪费。例如,一个数据库服务器可以使用多线程来处理多个客户端请求,而不是为每个请求创建一个独立的进程。

快的任务完成: 并行执行不同的任务可以加速任务的完成。多线程可以用于加速计算密集型任务、并行化数据处理、提高网络通信的效率等。

6.实现复杂性: 在某些情况下,使用多线程可以更容易地处理复杂的问题。例如,在模拟、游戏开发和科学计算中,多线程可以帮助分解复杂任务,简化代码。

7.提高系统稳定性: 多线程应用程序更容易实现错误恢复和故障隔离。如果一个线程崩溃,不会导致整个进程崩溃,从而提高了系统的稳定性。

线程的类型

用户级线程

什么是用户级线程

用户级线程(User-Level Threads)是在用户空间(用户程序的地址空间)中由线程库(Thread Library)提供支持的线程。这种线程的创建、调度和管理都在用户空间进行,而不需要内核(操作系统)的干预。用户级线程是与操作系统无关的,操作系统并不知道它们的存在。

在Java中,用户级线程通常是通过使用线程库或第三方框架来实现的,而不是使用标准的Thread类。一个常见的用户级线程库是协程库,如Quasar或Project Loom。这些库提供了在用户空间中实现用户级线程的功能
以下是使用Project Loom库创建用户级线程的示例代码:

import java.util.concurrent.ExecutionException;
import java.util.concurrent.Future;public class UserLevelThreadExample {public static void main(String[] args) {// 创建一个用户级线程Future<String> fiber = Fiber.schedule(() -> {return "Hello from a user-level thread!";});try {// 等待用户级线程执行完成并获取结果String result = fiber.get();System.out.println(result);} catch (ExecutionException | InterruptedException e) {e.printStackTrace();}}
}

优点

1.轻量级: 用户级线程相对于内核级线程来说更轻量级,因为它们不需要内核级别的上下文切换。线程的创建和销毁以及线程之间的切换都可以在用户空间中完成,这样的操作更快速、更高效。

2.独立调度: 用户级线程的调度不依赖于操作系统的调度算法,而是由线程库中的用户代码来决定。这使得开发者可以更灵活地控制线程的调度行为。

3.无需内核支持: 用户级线程不需要内核的支持,因此在一些特定场景下,可以更好地适应特定的需求。不依赖于内核,也意味着用户级线程可以在不同的操作系统上运行,而不需要修改。

4.适用于特定应用场景: 由于用户级线程的轻量级特性,它们通常用于特定的应用场景,例如需要快速响应、高度并发但不涉及复杂I/O操作的应用程序。

缺点

1.无法利用多核处理器: 由于用户级线程的调度是在用户空间完成的,无法充分利用多核处理器的并行性能。所有线程仍然运行在同一个核心上,无法实现真正的并行处理。

2.阻塞问题: 如果一个用户级线程在进行阻塞式I/O操作时,会导致整个进程的所有线程被阻塞,因为操作系统不知道线程的存在,无法将其他线程切换到运行态。

3.不稳定性: 用户级线程的稳定性受限于线程库的实现,不同的线程库可能存在不同的稳定性问题。

4.无法实现真正的并发控制: 用户级线程的并发控制受限于线程库,无法像内核级线程那样实现更高级别的并发控制。

内核级线程

什么是内核级线程

内核级线程(Kernel-Level Threads),也称为内核线程或系统线程,是由操作系统内核直接管理和调度的线程。这些线程在操作系统内核的支持下执行

在Java中,使用Thread类创建的线程通常是内核级线程。这是因为Thread类是Java的标准线程类,它直接映射到操作系统级别的线程。这些线程由Java虚拟机(JVM)管理,并受操作系统的调度和资源管理。

当使用Thread类创建线程时,JVM会将线程映射到底层操作系统的原生线程(通常是内核级线程),这允许Java应用程序利用多核处理器和操作系统的并发支持。这些线程具有操作系统级别的上下文切换,因此它们可以实现真正的并行执行,利用多核处理器的性能。

优点

1.多核支持: 内核级线程可以充分利用多核处理器的并行性,允许线程在不同的处理器核心上并行执行,提高了系统性能。

2.硬件抽象: 内核级线程提供了对硬件的抽象,使线程可以直接访问硬件资源,如文件系统、网络、设备驱动程序等。这使得内核级线程适用于需要直接与硬件交互的应用程序。

3.操作系统支持: 内核级线程依赖于操作系统提供的支持,因此可以利用操作系统的丰富功能,包括强大的进程调度、资源管理和安全性功能。

4.稳定性和隔离: 由于操作系统内核管理内核级线程,它们通常具有较高的稳定性和安全性。操作系统能够确保线程之间的隔离,从而防止一个线程的错误影响整个系统。

缺点

1.创建和销毁开销较大: 内核级线程的创建和销毁通常涉及较大的开销,因为它们需要操作系统内核的介入。这与用户级线程相比,用户级线程的创建和销毁开销较小。

2.线程切换开销: 内核级线程的切换通常涉及内核态到用户态的切换,这会引入较大的线程切换开销,包括上下文保存和恢复。这对于一些轻量级任务来说可能是昂贵的。

3.复杂性: 内核级线程的管理和调度由操作系统内核负责,因此具有更高的复杂性。这使得内核级线程的使用和调试可能相对复杂。

4.可伸缩性问题: 内核级线程的数量受限于操作系统内核的限制,因此当需要大量线程时,可能会遇到可伸缩性问题。

5.操作系统依赖性: 内核级线程的行为和性能在不同操作系统上可能会有所不同,因此它们对操作系统的依赖性较高。这可能会导致跨平台应用程序开发方面的挑战。

守护线程

什么是守护线程

守护线程是一种特殊类型的线程,它在后台运行,不会阻止程序的退出。当所有非守护线程结束时,守护线程会自动终止。它们通常用于执行后台任务,如垃圾回收。

使用场景

1.jvm 垃圾回收器
2.心跳检测

Java创建守护线程

        Thread thread = new Thread(new  ThreadDemo());//设置线程为守护线程thread.setDaemon(true);

线程的生命周期

1.新建(New): 当线程对象被创建但尚未启动时,线程处于新建状态。在这个阶段,线程对象已经被分配内存,但尚未开始执行。

2.就绪(Runnable): 线程处于就绪状态时,它已经被启动,但尚未分配到CPU执行时间片,等待操作系统的调度。在就绪状态下,线程可以随时开始执行。

3.运行(Running): 线程在运行状态时,它正在执行任务代码,占用CPU时间片。一个时刻只能有一个线程处于运行状态,即使有多个线程在就绪状态。

4.阻塞(Blocked): 当线程因某种原因而无法继续执行时,它进入阻塞状态。这些原因包括等待I/O操作完成、等待获取锁或等待其他资源。线程在阻塞状态下不会占用CPU时间,直到它可以继续执行。

5.等待(Waiting): 等待状态是阻塞状态的一种特殊情况。线程通常在等待某个条件满足时进入等待状态。例如,通过调用Object.wait()方法或Thread.sleep()方法。

6.定时等待(Timed Waiting): 定时等待状态是等待状态的一种变体,线程在等待一段指定的时间后会自动从定时等待状态恢复到就绪状态。例如,通过调用Thread.sleep()方法指定等待时间。

7.终止(Terminated): 线程的生命周期在终止状态下结束。线程可以因任务执行完毕、异常终止或手动终止而进入终止状态。一旦线程进入终止状态,它不能再切换到其他状态。

Java创建线程的方式

继承Thread

public class ThreadDemo extends Thread  {@Overridepublic void run() {System.out.println("ThreadDemo start");}
}
public class ThreadMain {public static void main(String[] args) {System.out.println("ThreadMain start");Thread thread = new Thread(new ThreadDemo());//当调用thread.start方法的时候会执行ThreadDemo 的run方法打印ThreadDemo startthread.start();}
}

实现Runnable接口

public class RunnableDemo implements Runnable{@Overridepublic void run() {System.out.println("RunnableDemo start");}
}
    public static void main(String[] args) {System.out.println("ThreadMain start");Thread thread = new Thread(new RunnableDemo());//当调用thread.start方法的时候会执行RunnableDemo 的run方法打印RunnableDemo startthread.start();}

实现Callable接口

因为Thread对象的参数不接收Callable对象,所以要定义一个既可以接收Callable对象的,又可以被Thread对象接收的对象------->FutureTask对象。FutureTask调用时会调用run方法,而FutureTask类中又定义的有Callable对象,所以会调用call()方法。

public class CallableDemo {public static void main(String[] args) throws InterruptedException, ExecutionException {//Callable接口的实现类:不同数据范围的计算任务SumCalu sumCalu1 = new SumCalu(1,300);//1-300SumCalu sumCalu2 = new SumCalu(301,500);//301-500SumCalu sumCalu3 = new SumCalu(501,1000);//501-1000//FutureTask类间接也是Callable的实现类FutureTask<Integer> futureTask1 = new FutureTask<Integer>(sumCalu1);FutureTask<Integer> futureTask2 = new FutureTask<Integer>(sumCalu2);FutureTask<Integer> futureTask3 = new FutureTask<Integer>(sumCalu3);//创建线程对象,传入Thread t1 = new Thread(futureTask1);Thread t2 = new Thread(futureTask2);Thread t3 = new Thread(futureTask3);//启动线程t1.start();t2.start();t3.start();//线程执行结束,分别获取各自线程的返回结果System.out.println("开始分别获取......");Integer sum1 = futureTask1.get();Integer sum2 = futureTask2.get();Integer sum3 = futureTask3.get();//汇总结果System.out.println("汇总各自计算结果");Integer sum = sum1+sum2+sum3;System.out.println("汇总结果:"+sum);}
}//通过Callable接口实现类SumCalu封装某个范围内数据的累加和
class SumCalu implements Callable<Integer>{private int begin,end;//有参构造方法public SumCalu(int begin,int end) {this.begin = begin;this.end = begin;}//执行逻辑@Overridepublic Integer call() throws Exception {int total = 0;for(int i=begin;i<=end;i++) {total += i;}//获取当前线程的名称System.out.println(Thread.currentThread().getName());return total;}}

通过线程池创建

执行线程任务有两种方式:executer()和submit()。execute()只能提交Runnable类型的任务,且没有返回结果;submit()既能提交Runnable类型的任务,又能提交Callable类型的任务,且可以返回Future类型的结果。

public class ExecutorDemo {public static void main(String[] args) {//创建固定数量(10)的线程池ExecutorService executorService =  Executors.newFixedThreadPool(10);//不确定数量的线程池请求while(true) {//向线程池提交一个执行任务(Runnable接口实现类对象)//线程池分配一个"空闲线程"执行该任务//如果没有空闲线程,则该任务进入"等待队列(工作队列)"executorService.execute(new Runnable() {@Overridepublic void run() {System.out.println(Thread.currentThread().getName()+"执行了一次任务!");try {//当前线程休眠1000毫秒Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}}});}}
}

引入多线程导致的问题

1.死锁(Deadlocks): 死锁是多个线程相互等待对方释放资源的情况,导致所有线程无法继续执行。死锁的预防和解决需要小心的资源分配和死锁检测。

2.上下文切换开销: 频繁的线程上下文切换可能导致性能下降,因为上下文切换本身需要时间和资源。合理设计线程数量和调度策略可以减少这种开销。

3.线程安全问题: 多线程应用可能涉及到共享数据,开发者需要确保多线程操作不会导致数据损坏或不一致。这包括正确使用锁、原子操作和线程安全的数据结构。

线程上下文切换

什么是线程的上下文切换

线程的上下文切换是指在多线程环境下,操作系统或线程调度器暂停当前执行的线程,并将其执行上下文(包括寄存器状态、程序计数器、栈指针等)保存到内存中,然后选择另一个线程来执行。这个过程可以理解为从一个线程的执行状态切换到另一个线程的执行状态,以便并发执行多个线程。

线程为什么需要上下文切换

1.多任务并发执行: 在多线程应用中,多个线程共享同一个CPU。为了实现并发执行,操作系统需要在不同的线程之间进行切换,以确保它们都有机会执行。上下文切换是实现这种并发的方式之一。

2.抢占式调度: 在多线程环境中,操作系统通常会采用抢占式调度策略。这意味着更高优先级的线程可以在任何时刻抢占 CPU 时间片,以确保紧急任务能够立即执行。上下文切换是在这些抢占操作中的一部分,以实现线程的切换。

3.阻塞和等待: 当线程执行阻塞操作,如等待I/O完成、等待锁或等待条件变量满足时,它会被暂停,以便其他线程可以执行。一旦阻塞操作完成,线程需要从之前的位置继续执行,这需要上下文切换。

4.线程间通信和同步: 在多线程应用中,线程之间需要协调和同步操作,以避免竞争条件和数据访问冲突。上下文切换允许线程在不同的时间点执行,以进行协调和同步操作。

5.多核处理器利用: 在多核处理器中,多个线程可以并行执行,以充分利用硬件资源。上下文切换允许不同的线程在不同的核心上执行,以实现并行性。

引发上下文切换的条件

1.抢占式调度: 当操作系统内核决定暂停当前运行的线程,并将 CPU 时间片分配给另一个线程时,发生抢占式的上下文切换。这通常基于线程的优先级和调度策略,以确保各个线程都有公平的机会执行。

2.阻塞和等待: 当一个线程执行阻塞操作(如等待I/O操作、获取锁、等待条件满足等)时,它会被暂停,并让出 CPU。一旦阻塞操作完成,线程会重新进入可运行状态,并可能被调度执行。

3.线程主动让出 CPU: 线程可以通过调用 yield() 或 sleep() 等方法来显式地让出 CPU,以触发上下文切换。

频繁上下文切换的成本及代价

成本

时间成本: 上下文切换需要保存当前线程的执行上下文,并加载新线程的执行上下文。这些操作需要一定的时间,可能会导致延迟。

资源成本: 上下文切换需要分配和管理内存资源,包括保存和恢复执行上下文所需的数据结构。这可能占用额外的内存和操作系统资源。

导致的问题

性能下降: 过多的上下文切换会导致性能下降。这是因为线程上下文切换本身需要时间,而且它可能会导致CPU缓存失效,从而降低了执行效率。在CPU密集型应用中,频繁的上下文切换可能会降低应用的整体性能。

竞争条件: 上下文切换时,线程状态可能会被暂停,这可能会导致竞争条件。竞争条件是一种在多线程应用中可能导致数据不一致或错误的情况。

死锁: 不正确的线程同步和阻塞操作可能导致死锁。如果线程在等待资源时被上下文切换,而其他线程持有它需要的资源,可能会导致死锁。

资源争用: 上下文切换可能导致线程争用系统资源,如锁、内存、文件句柄等。这可能会导致资源瓶颈和性能问题。

不公平调度: 不恰当的线程调度策略可能导致某些线程获得更多的CPU时间,而其他线程受到忽视。这可能会降低系统的公平性。

线程安全问题

什么是线程安全问题

线程安全问题是指在多线程环境下,当多个线程同时访问和修改共享的数据或资源时可能导致的问题。这些问题包括数据竞争、不一致的状态和其他可能破坏应用程序正确性的情况。

导致线程不安全问题的原因

1.竞争条件(Race Conditions): 多个线程尝试同时访问和修改共享数据,但没有适当的同步机制来保护数据的完整性。这可能导致不确定的结果。
2.未加锁的访问: 如果多个线程同时访问共享数据而没有获取正确的锁,可能会导致数据不一致性。例如,一个线程可能在另一个线程修改数据时读取数据。
3.共享可变状态: 多个线程共享可变状态(例如对象的属性或全局变量),而没有适当的同步机制。一个线程可能会在另一个线程尚未完成修改之前读取或修改数据。
4.不一致的内存访问: 多核处理器上,不同的CPU核心可能会访问不一致的内存副本,导致线程间的数据不一致性。这通常需要使用内存屏障和同步操作来解决。
5.不同线程的执行顺序: 线程的执行顺序是不确定的,因此某些操作可能在不同线程中以不同的顺序发生,导致问题。

如何解决线程不安全问题

1.使用锁机制: 最常见的方法是使用锁,如互斥锁(Mutex)或信号量(Semaphore),以确保在任何时刻只有一个线程可以访问共享资源。通过获取锁,线程可以互斥地执行关键代码段,从而避免竞争条件。
2.使用原子操作: 原子操作是不可分割的操作,它们保证在单个操作中执行多个线程之间的操作。编程语言和库通常提供原子操作,如原子递增或原子交换,以确保线程安全。

死锁

形成死锁的原因

它发生在多个线程或进程之间相互等待对方释放资源的情况下,导致所有线程或进程无法继续执行,形成了僵局。
典型的死锁情况涉及多个线程或进程,每个线程或进程都在等待其他线程或进程释放它们占用的资源。这种等待是很持久的,因为每个线程都无法继续执行,直到它所需的资源被释放。

A线程在等待B线程释放锁 B线程在等待A线程释放锁 然后谁都获取不到对方的锁 直接造成死锁

导致死锁的条件

  1. 互斥条件:所谓互斥就是进程在某一时间内独占资源。
  2. 请求与保持条件:一个进程因请求资源而阻塞时,对已获得的资源保持不放。
  3. 不剥夺条件:进程已获得资源,在末使用完之前,不能强行剥夺。
  4. 循环等待条件:若干进程之间形成一种头尾相接的循环等待资源关系。

排查死锁问题案例

package com.alipay.alibabademo.rocketmq;public class ThreadMain {public static void main(String[] args) {final Object lock1 = new Object();final Object lock2 = new Object();Thread thread1 = new Thread(() -> {synchronized (lock1) {System.out.println("Thread 1: Holding lock1...");try { Thread.sleep(100); } catch (InterruptedException e) {}System.out.println("Thread 1: Waiting for lock2...");synchronized (lock2) {System.out.println("Thread 1: Acquired lock2!");}}});Thread thread2 = new Thread(() -> {synchronized (lock2) {System.out.println("Thread 2: Holding lock2...");try { Thread.sleep(100); } catch (InterruptedException e) {}System.out.println("Thread 2: Waiting for lock1...");synchronized (lock1) {System.out.println("Thread 2: Acquired lock1!");}}});thread1.start();thread2.start();// 等待两个线程执行完成try {thread1.join();thread2.join();} catch (InterruptedException e) {e.printStackTrace();}}
}

在这里插入图片描述
1.通过jps命令得到正在运行的Java程序的PID
2.通过通过 jstack查看这个PID下的日志
如图可以看出
Thread-0" 线程正在等待锁 <0x0000000716e1f848>,并且已经锁定锁 <0x0000000716e1f838>。
“Thread-1” 线程正在等待锁 <0x0000000716e1f838>,并且已经锁定锁 <0x0000000716e1f848>。
这种情况下,“Thread-0” 持有锁 <0x0000000716e1f848>,但它需要锁 <0x0000000716e1f838> 来继续执行。同时,“Thread-1” 持有锁 <0x0000000716e1f838>,但它需要锁 <0x0000000716e1f848> 来继续执行。由于它们相互等待对方释放锁,导致了死锁。

多线程的三大特性

1.原子性(Atomicity): 原子性是指一个操作是不可中断的,要么全部执行,要么都不执行。在多线程编程中,原子操作是线程安全的,多个线程同时执行原子操作不会导致数据不一致或竞态条件。Java 提供了一些原子操作的机制,如 synchronized 关键字、java.util.concurrent.atomic 包中的原子类等,用于确保原子性。

2.可见性(Visibility): 可见性指的是一个线程对共享变量的修改能够被其他线程立即看到。在多核处理器和多线程环境中,由于缓存和指令重排等因素,共享变量的可见性问题可能会导致线程不一致。为了确保可见性,Java 提供了 volatile 关键字和锁机制(如 synchronized)来同步线程之间的内存访问。

3.有序性(Ordering): 有序性指的是线程执行操作的顺序要与程序中的顺序一致。在多线程环境中,编译器和处理器可能对指令进行重排,但这些重排不能改变程序的原始语义。Java 通过 volatile 和锁机制来确保有序性。

线程休眠(Sleep)

sleep 是一个在多线程编程中常用的方法,它用于让线程休眠(暂停执行)一段指定的时间。 但在休眠期间,它不会释放已经持有的锁,并且会放弃 CPU 的执行权。

try {// 让当前线程休眠500毫秒(半秒)Thread.sleep(500);
} catch (InterruptedException e) {// 处理中断异常
}

上述代码会让当前线程休眠500毫秒。在这段时间内,线程不会执行任何任务,然后它会自动唤醒并继续执行。

线程控制(yield)

yield 是一个线程控制的方法,用于暗示当前线程愿意放弃 CPU 的执行权,使其他具有相同或更高优先级的线程有机会运行。但它不会释放锁,只是放弃 CPU 的执行权。

Thread.yield();

yield 方法的调用会导致当前线程从运行状态切换到就绪状态,让操作系统的线程调度器决定下一个运行的线程。通常情况下,如果没有更高优先级的线程需要执行,yield 方法可能会让当前线程继续运行。

Join

Join用于等待另一个线程执行完成。当一个线程调用另一个线程的 join 方法时,它会等待目标线程执行完毕后再继续执行。这通常用于协调多个线程的执行顺序。

Thread threadToJoin = new Thread(() -> {// 线程执行的任务
});// 启动目标线程
threadToJoin.start();try {// 等待目标线程执行完毕threadToJoin.join();
} catch (InterruptedException e) {// 处理中断异常
}

在上述代码中,主线程启动了一个目标线程 threadToJoin,然后通过 threadToJoin.join() 来等待 threadToJoin 线程执行完毕。一旦 threadToJoin 线程执行完成,主线程才会继续执行。

线程阻塞(Wait)

当一个线程调用 wait 方法时,它会释放之前获得的锁,并进入等待状态。在等待期间,线程不会占用 CPU 的执行权,它会让出 CPU 给其他线程执行。只有当另一个线程通过 notify 或 notifyAll 方法通知等待的线程条件已满足时,等待的线程才会被唤醒,重新尝试获取锁,并继续执行。

synchronized (lockObject) {while (conditionIsNotMet) {try {lockObject.wait(); // 当条件不满足时,线程释放锁并等待} catch (InterruptedException e) {// 处理中断异常}}// 执行线程需要的操作
}

上述代码中,wait 方法通常在一个同步块内部使用,以确保线程对共享资源的访问是同步的。while 循环用于检查某个条件是否满足,如果条件不满足,线程调用 wait 方法释放锁并等待,直到另一个线程通过 notify 或 notifyAll 方法通知它条件已经满足。一旦线程被唤醒,它会重新尝试获取锁并继续执行

线程唤醒

notify

notify 是 Java 多线程编程中用于线程通信和同步的方法之一,它用于唤醒等待在对象监视器上的一个线程。当一个线程调用某个对象的 notify 方法时,它会通知等待在这个对象上的某个线程,使其从等待状态进入就绪状态,然后等待 CPU 调度执行。

notifyAll

它用于唤醒等待在对象监视器上的所有线程。当一个线程调用某个对象的 notifyAll 方法时,它会通知所有等待在这个对象上的线程,使它们从等待状态进入就绪状态,然后等待 CPU 调度执行。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/103241.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Python高效实现网站数据挖掘

在当今互联网时代&#xff0c;SEO对于网站的成功至关重要。而Python爬虫作为一种强大的工具&#xff0c;为网站SEO带来了革命性的改变。通过利用Python爬虫&#xff0c;我们可以高效地实现网站数据挖掘和关键词分析&#xff0c;从而优化网站的SEO策略。本文将为您详细介绍如何利…

新风机缺点有哪些?

虽然新风机在提供新鲜空气和改善室内空气质量方面有很多优点&#xff0c;但它也存在一些缺点。下面列举几个常见的新风机缺点&#xff1a; 安装成本较高&#xff1a;新风机需要通过管道连接室内和室外&#xff0c;需要对房屋进行改造和安装。这可能会增加一些额外的安装成本&am…

代码随想录算法训练营第六十二、六十三天 | 单调栈 part 2 | 503.下一个更大元素II 、42. 接雨水、84.柱状图中最大的矩形

目录 503.下一个更大元素II思路代码 42. 接雨水思路一 双指针思路二 单调栈代码 84.柱状图中最大的矩形思路一 双指针思路二 单调栈代码 503.下一个更大元素II Leetcode 思路 将数组乘2来遍历即可&#xff0c;就是加长版的每日温度。 但是处理起来会有细节&#xff0c;如果…

JAVA学习第一天,java的运行方式

对未来很迷茫&#xff0c;不知道以后能出去干什么&#xff0c;好像掌握的东西很少&#xff0c;从今天开始学习学习java吧&#xff0c;让自己充实起来&#xff0c;记录一下。 jav…

Maven系列第6篇:生命周期和插件详解?

maven系列目标&#xff1a;从入门开始开始掌握一个高级开发所需要的maven技能。 这是maven系列第6篇。 整个maven系列的内容前后是有依赖的&#xff0c;如果之前没有接触过maven&#xff0c;建议从第一篇看起&#xff0c;本文尾部有maven完整系列的连接。 前面我们使用maven…

刷新页面,数据丢失

刷新页面数据丢失原因很多&#xff0c;其中有一种是解析错误&#xff0c;没有解构出来。 报错内容如下(类似这个报错)&#xff1a; 数据结构如下&#xff1a; this.$router.push({name: DetailComparison,query: {rowDetail: rowDetail || null} }) 修改数据结构&#xff1a…

OpenAI科学家谈GPT-4的潜力与挑战

OpenAI Research Scientist Hyung Won Chung 在首尔国立大学发表的一场演讲。 模型足够大&#xff0c;某些能力才会显现&#xff0c;GPT-4 即将超越拐点并在其能力上实现显着跳跃。GPT-3 和 GPT-4 之间的能力仍然存在显着差距&#xff0c;并且尝试弥合与当前模型的差距可能是无…

使用eBPF加速阿里云服务网格ASM

背景 随着云原生应用架构的快速发展&#xff0c;微服务架构已经成为了构建现代应用的主要方式之一。而在微服务架构中&#xff0c;服务间的通信变得至关重要。为了实现弹性和可伸缩性&#xff0c;许多组织开始采用服务网格技术来管理服务之间的通信。 Istio作为目前最受欢迎的…

C++基础之类二(类的实例化,This指针)

目录 类的实例化 概念 类和对象的区别 计算一个类 不同的类的大小 类的存储模型 内存对齐 让结构体按照指定的默认对齐数进行对齐 确定偏移量 大小端字节序 This指针 概念 this指针四大特性 一些关于this指针的问题 总结 之前学过了类&#xff0c;但在编程中类无法直接使用&…

鼎鑫鸿鄴利用国家的策略优势和满足全球需求并驾齐驱

随着全球对可再生能源的需求不断增长&#xff0c;鼎鑫鸿鄴新能源科技有限公司正充分利用中国政府的策略优势&#xff0c;积极满足全球能源使用需求&#xff0c;并在可再生能源领域崭露头角。中国属于全球最大的太阳能市场&#xff0c;在技术研发、产能建设和市场规模等方面拥有…

[Spring] SpringMVC 简介(一)

目录 一、SpringMVC 简介 1、什么是 MVC 2、什么是 SpringMVC 3、SpringMVC 实现原理 4、SpringMVC 的特点 二、简单案例 1、引入依赖 2、在 web.xml 中配置前端控制器 DispatcherServlet 3、创建 SpringMVC 的配置文件 4、创建请求控制器 5、测试页面 6、访问不到 …

数学建模——人工神经网络模型

一、人工神经网络简介 1、神经网络起源与应用 1943年心理学家McCulloch和数学家Pitts提出神经元生物数学模型&#xff08;M-P模型&#xff09;&#xff0c;后来人工神经网络(Artifical Neural Network,ANN)是在生物神经网络(Biological Neural Network,BNN)基础上发展起来的&a…

MobaXterm使sftp目录与cmd目录同步

MobaXterm使sftp目录与cmd目录同步 创建session时在ssh菜单栏中选择Advanced SSH settings其中SSH-browser type 选择SCP&#xff08;enhanced speed&#xff09; 勾选Follow SSH path

TS(五):装饰器

装饰器 启用装饰器支持类装饰器定义并使用装饰器对比不使用装饰器装饰器叠加实现消息提示统一响应装饰器工厂 方法装饰器登录状态验证数据请求 属性装饰器大小写转换 元数据安装依赖基础用法 参数装饰器验证参数是否为必填项 启用装饰器支持 修改 tsconfig.json {"exper…

和鲸 ModelWhale 与华为 OceanStor 2910 计算型存储完成兼容性测试

数智化时代&#xff0c;数据总量的爆炸性增长伴随着人工智能、云计算等技术的发展&#xff0c;加速催化了公众对于数据存储与应用的多元化需求。同时&#xff0c;数据也是重要的基础资源和战略资源&#xff0c;需要严格保障其安全性、完整性。搭建国产数据基础设施底座&#xf…

信息系统项目管理师第四版学习笔记——高级项目管理

项目集管理 项目集管理角色和职责 在项目集管理中涉及的相关角色主要包括&#xff1a;项目集发起人、项目集指导委员会、项目集经理、其他影响项目集的干系人。 项目集发起人和收益人是负责承诺将组织的资源应用于项目集&#xff0c;并致力于使项目集取得成功的人。 项目集…

实验2.1.2 交换机的常用配置

项目2 交换技术的位置 活动2 交换机的常用配置 一、具体要求&#xff1a; &#xff08;1&#xff09;添加1台计算机&#xff0c;将标签名更改为PC1。 &#xff08;2&#xff09;添加1台S3700-26C-HI交换机&#xff0c;标签名为SWA&#xff0c;将交换机的名称设置为SWA。 &am…

华为云云耀云服务器L实例评测 | 实例评测使用之硬件参数评测:华为云云耀云服务器下的 Linux 磁盘目录分析神器 ncdu

华为云云耀云服务器L实例评测 &#xff5c; 实例评测使用之硬件参数评测&#xff1a;华为云云耀云服务器下的 Linux 磁盘目录分析神器 ncdu 介绍华为云云耀云服务器 华为云云耀云服务器 &#xff08;目前已经全新升级为 华为云云耀云服务器L实例&#xff09; 华为云云耀云服务器…

关键词搜索快手商品列表数据,快手商品列表数据接口,快手API接口

在网页抓取方面&#xff0c;可以使用 Python、Java 等编程语言编写程序&#xff0c;通过模拟 HTTP 请求&#xff0c;获取快手网站上的商品页面。在数据提取方面&#xff0c;可以使用正则表达式、XPath 等方式从 HTML 代码中提取出有用的信息。值得注意的是&#xff0c;快手网站…

Android Studio: unrecognized Attribute name MODULE

错误完整代码&#xff1a; &#xfffd;&#xfffd;&#xfffd;&#xfffd;&#xfffd;&#xfffd; (1.8.0_291) &#xfffd;г&#xfffd;&#xfffd;&#xfffd;&#xfffd;쳣&#xfffd;&#xfffd;&#xfffd;&#xfffd;&#xfffd;&#xfffd;&#xff…