Java并发编程-线程基础

news2025/1/9 20:12:43
  • 哈喽,大家好,这篇文章主要讲解Java并发编程,线程基础的知识点,讲解知识点的同时也会穿插面试题的讲解.

主要讲解以下内容

  • 进程,线程基础常见的一些区别比较
  • 如何使用和查看线程
  • 理解线程如何运行以及线程上下文切换的知识
  • 线程方法
  • 线程状态

希望能给大家带来帮助.加油~~~

目录

进程,线程基础常见的一些区别比较

进程与线程的区别

并发与并行的区别

Java中守护线程和用户线程的区别?

​编辑

说下同步、异步、阻塞和非阻塞

JDK19中的虚拟线程了解么? /了解协程么?

线程的使用与查看

线程创建的方式有哪些?/Java中线程的实现方式?

继承Thread类

普通类继承Thread类

匿名内部类

继承Thread类的缺点

实现Runnable接口

普通类实现Runnable接口

匿名Runnable实现类

使用lambda表达式

实现Runnable接口的缺点

实现Callable接口创建线程

普通类实现Callable接口

匿名callable实现类

实现Callable接口来创建线程分析

使用线程池创建线程

Runnable与Callable的区别?

Thread 与 Runnable 的关系

线程类的构造方法,静态块是被哪个线程调用的?

cpu占用过高的分析思路

如何查看进程线程的方法

线程上下文切换/如何理解线程的运行

栈与栈帧

 多线程运行的原理

多线程运行原理

什么是线程上下文切换

发生上下文切换的原因

线程数越多越好么 ?

线程方法

start与run

start与run的区别

为什么调用start()方法时会执行run()方法,而不直接执行run()方法?

为什么start方法不能重复调用?而run方法却可以?

可以直接调用 Thread 类的 run 方法吗?

sleep,yield,wait

sleep与yield的区别

sleep与wait的区别

为什么 wait() 方法不定义在 Thread 中?

为什么 sleep() 方法定义在 Thread 中?

线程的优先级

Thread.sleep(0)的作用是什么?

join方法

interrupt方法

如何停止一个正在运行的线程? /如何优雅的关闭线程

错误停止线程的方式

interrupt, isInterrupted, interrupted的区别是什么

notify和notifyAll有什么区别

线程的状态

线程的5种状态

线程的六种状态

线程状态之间的变化

线程死亡的几种情况

线程阻塞的几种情况


进程,线程基础常见的一些区别比较

进程与线程的区别

什么是进程 ? 

比如我们打开一个程序,就相当于开启一个进程,我们知道程序是由指令和数据构成的,这些指令要运行,数据要读写就必须将指令加载到CPU,数据加载到内存中,指令运行的过程中需要用到磁盘,网络等设备,进程就是用来加载指令,管理内存,管理IO的.

总结一句话 : 当一个程序被运行,从磁盘加载这个程序的代码到内存,这时就开启了一个进程.

什么是线程 ? 

个线程就是一个指令流,将指令流中的一条条指令以一定的顺序交给CPU执行.

一个进程包含了多个线程,Java 中,线程作为最小调度单位,进程作为资源分配的最小单位。

总结一下进程与线程的区别:

  • 进程是资源分配的基本单位,线程是程序执行/系统调度的基本单位;
  • 进程是正在运行程序的实例,进程中包含了线程,每个线程执行不同的任务
  • 进程拥有独立的内存空间和资源,而线程则共享进程的内存和资源
  • 进程之间的通信比较复杂,而线程通信相对简单,他们共享进程中的资源
  • 线程更加轻量,线程的上下文切换成本一般要比进程的上下文切换低

并发与并行的区别

对于单核CPU : 线程实际上还是串行执行的.

由于操作系统有一个组件叫做任务调度器,他会将CPU的时间片分给不同的程序使用,只是由于CPU在线程间的切换速度非常快,所以我们人总感觉是同时执行的.

即 : 微观串行,宏观并行.

我们把 对于线程轮流使用CPU的这种做法叫做并发.

在多核CPU下,才能真正的可以做到并行执行,也就是多个CPU核心都可以调度运行线程.

总结一下 :

  • 并发是同一时间应对多件事情的能力,比如多个线程轮流使用一个或者多个CPU
  • 并行是同一时间动手做多件事情的能力,比如4核CPU同时执行4个线程

现在大多数都是多核CPU,在多核CPU下 , 没有绝对的并发和绝对的并行,大多数情况下都是既有并发又有并行.

Java中守护线程和用户线程的区别?

用户线程 : 默认情况下我们创建的线程或线程池都是用户线程,所以用户线程也被称之为普通线程。
守护线程 : 守护线程是为用户线程服务的,一般是指后台线程. 

默认情况下,Java 进程需要等待所有线程都运行结束,才会结束。有一种特殊的线程叫做守护线程,只要其它非守护线程运行结束了,即使守护线程的代码没有执行完,也会强制结束。当程序中的用户线程全部执行结束之后,守护线程也会结束.

  • 垃圾回收器线程:  垃圾回收就是会将堆中的没有被引用的对象回收掉,当程序停止的时候,垃圾回收线程也会停止
  • 守护线程的设置 setDaemon(true) 必须要放在线程的 start() 之前,否则程序会报错。也就是说在运行线程之前,一定要先确定线程的类型,并且线程运行之后是不允许修改线程的类型的。
@Slf4j(topic ="c.TestDemo1")
public class TestDemo1 {
    public static void main(String[] args) throws InterruptedException {
        log.debug("开始运行...");
        Thread t1 = new Thread(() -> {
            log.debug("开始运行...");
            try {
                sleep(2);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            log.debug("运行结束...");
        }, "daemon");

        // 设置该线程为守护线程
        t1.setDaemon(true);
        t1.start();
        sleep(1);
        log.debug("运行结束...");
    }
}

说下同步、异步、阻塞和非阻塞

JDK19中的虚拟线程了解么? /了解协程么?

线程的使用与查看

线程创建的方式有哪些?/Java中线程的实现方式?

线程的创建/实现方式有四种

  • 继承Thread类
    • 普通类继承Thread类
    • 匿名内部类
  • 实现Runnable接口
    • 普通类实现Runnable接口
    • 匿名Runnable实现类
    • 使用lambda表达式
  • 实现Callable接口
    • 普通类实现Callable接口
    • 匿名callable实现类
  • 使用线程池创建线程

继承Thread类

普通类继承Thread类

//使用普通类继承Thread
class MyThread extends Thread{
    @Override
    public void run() {
        System.out.println("我使用的是普通类来继承Thread类");
    }
}
public class ThreadDemo1 {
    public static void main(String[] args) {
        Thread t = new MyThread();
        t.start();
    }
}

匿名内部类

public class ThreadDemo2 {
    //使用匿名内部类的方式创建线程
    public static void main(String[] args) {
        Thread t = new Thread(){
            @Override
            public void run() {
                System.out.println("我是run方法");
            }
        };
        //创建线程
        t.start();
    }
}

继承Thread类的缺点

  • 使用继承Thread类的方式创建新线程,最大的局限就是不支持多继承,因为Java语言的特点就是单继承.继承了Thread类就不能继承其他类了

实现Runnable接口

普通类实现Runnable接口

class MyRunnable implements Runnable{
    @Override
    public void run() {
        System.out.println("我使用普通类实现Runnable接口来创建线程");
    }
}
public class ThreadDemo3 {
    public static void main(String[] args) {
        MyRunnable myRunnable = new MyRunnable();
        Thread t = new Thread(myRunnable);
        t.start();
    }
}

匿名Runnable实现类

public static void main(String[] args) {
    //使用匿名的Runnable实现类
    Thread t = new Thread(new Runnable() {
        @Override
        public void run() {
            System.out.println("我是run方法");
        }
    });
    //启动线程
    t.start();
}

使用lambda表达式

public static void main(String[] args) {
    //使用Lambda表达式
    Thread t = new Thread(()->{
        System.out.println("我是Lambda创建的线程");
    });
    //启动线程
    t.start();
}

实现Runnable接口的缺点

实现Runnable接口和继承Thread类的缺点都是不能获取线程执行的结果.但是如果是在不要求得到线程执行的返回结果的情况下就使用Lambda表达式最简洁

FutureTask也要需要与callable进行配合使用,futureTask参数要加上callable

实现Callable接口创建线程

普通类实现Callable接口

class MyCallable implements Callable<Integer> {
    @Override
    public Integer call() throws Exception {
        int sum = 0;
        for(int i =0;i<100;++i){
            sum += i;
        }
        return sum;
    }
}
public class ThreadDemo6 {
    public static void main(String[] args) throws ExecutionException, InterruptedException {
        MyCallable callable = new MyCallable();
        FutureTask<Integer> futureTask = new FutureTask<>(callable);
        Thread t = new Thread(futureTask);
        t.start();
        int res = futureTask.get();
        System.out.println(res);
    }
}

匿名callable实现类

public static void main(String[] args) throws ExecutionException, InterruptedException {
    //使用匿名callable实现类
    FutureTask<Integer> futureTask = new FutureTask<>(new Callable<Integer>() {
        @Override
        public Integer call() throws Exception {
            int sum = 0;
            for(int i =0;i<100;++i){
                sum += i;
            }
            return sum;
        }
    });
    Thread t = new Thread(futureTask);
    t.start();
    int res = futureTask.get();
    System.out.println(res);
}

实现Callable接口来创建线程分析

callable接口要作为参数放到FutureTask对象中.

根据源码分析 FutureTask 是间接实现了 Runnable接口,所以FutureTask也可以作为一个任务对象,他还有一个future接口,这个接口可以用于得到线程执行的结果,通过futureTask.get()就可以获取到线程执行结果交给其他线程来使用. 而对于Runnable是不能得到线程执行的结果

  • 首先要创建好Callable实例(或者使用匿名内部类的方式),将Callable实例传入FutureTask的构造方法参数中
  • 然后创建线程对象,将FutureTask对象传入到线程对象构造参数中.
  • 当启动线程后,就会执行callable的call方法,然后将结果返回给FutureTask对象保存起来
  • 然后线程执行完毕之后,可以通过futureTask.get()得到线程执行的结果

如果需要线程的执行结果,那就可以使用实现Callable接口的方式

使用线程池创建线程

//使用线程池来创建线程
ExecutorService executorService = Executors.newFixedThreadPool(10);
executorService.submit(new Runnable() {
    @Override
    public void run() {
        System.out.println("使用线程池来创建线程!!!");
    }
});

Runnable与Callable的区别?

首先实现Runnable接口和实现Callable接口都可以创建线程

两者的区别 :

  • Callable的任务执行后有返回值,而Runnable的任务是没有返回值的. , 运行Callable任务可以拿到一个Future对象, 可以得到异步计算的结果. 对于Runnable不能获取线程的执行结果
  • Callable规定(重写)的方法是call(),Runnable规定(重写)的方法是run()
  • Callable接口的call() 方法允许抛出异常;而Runnable接口的run方法的异常只能内部消化,不能继续上抛.

Thread 与 Runnable 的关系

Thread t1 = new Thread(() ->{
    System.out.println("xxxx");
}
);
Runnable runnable = ()->{
    System.out.println("xxx");
};
Thread t2 = new Thread(runnable);
  • 通过源码我们可以发现,他们最终都是调用的是run方法,有Runnable就调用Runnable的run方法
  • 方法1 是把线程和任务合并在了一起,方法2 是把线程和任务分开了
  • Runnable 更容易与线程池等高级 API 配合
  • Runnable 让任务类脱离了 Thread 继承体系,更灵活 ,组合优于继承

线程类的构造方法,静态块是被哪个线程调用的?

线程类的构造方法和静态块是由主线程调用的。

在Java程序中,当我们创建一个线程对象时,实际上是在主线程中创建了一个新的线程对象。这个新线程的构造方法和静态块都是在主线程中被调用执行的。

具体来说,在主线程中创建线程对象时,会调用线程类的构造方法来初始化线程对象,并执行构造方法中的逻辑。如果线程类中有静态块,主线程在加载这个类的时候会执行静态块中的代码,完成静态资源的初始化工作。

一旦创建完线程对象,我们可以通过调用start()方法启动线程,此时主线程会调用新线程的run()方法,新线程开始独立执行,并与主线程并发运行。

cpu占用过高的分析思路

  • top -H -p 进程id  -->查看进程中的线程信息 ,先使用top命令找出CPU占比最高的 到底是哪一个进程CPU过高
  • 查看线程信息 : ps H -eo pid,tid,%cpu | grep 进程pid 进一步知道进程中的哪一个线程CPU过高

  • 可以使用jstack 进程id->pid  得到该进程的所有线程栈的信息 ->线程的具体信息

  • 因为上一步之前得到的tid是10进制的,所以需要把cpu过高的那个线程的tid转换成16进制,在线程栈信息中找该16进制的tid

如何查看进程线程的方法

windows
  • 任务管理器可以查看进程和线程数,也可以用来杀死进程
  • tasklist 查看进程
  • taskkill 杀死进程
linux
  • ps -ef | grep 进程名称
  • 查看该进程及其所有子线程:ps -efL|grep PID

  • kill 杀死进程
  • 用于查看系统中所有线程,比如:top -H

  • top -H -p <PID> 查看某个进程(PID)的所有线程
Java
  • jps 命令查看所有 Java 进程,以及进程pid
  • jstack <PID> 查看某个 Java 进程(PID)的所有线程状态
  • jconsole 来查看某个 Java 进程中线程的运行情况(图形界面)

线程上下文切换/如何理解线程的运行

先理解栈帧的概念,我们一步一步了解线程的运行原理

栈与栈帧

JVM是由堆,栈,方法区所组成.我们的栈内存其实就是给我们线程来使用的.每一个线程启动后虚拟机就会为其分配一块栈内存

每一个栈是由多个栈帧(Frame)组成的,栈帧就是一次方法调用所占用的内存

每一个线程只能有一个活动栈帧,就是对应着当前正在执行的方法

每一个栈帧对应一个方法的调用

我们启动main方法的时候==>对应着就启动一个主线程,主线程一启动,虚拟机就会为主线程分配一块栈内存(由许多栈帧组成),调用main方法就会为main方法产生一个栈帧

在调用方法就会对其方法在产生一个栈帧(压栈),方法执行完之后,栈内存就会被释放,同时调用的时候记录返回地址,执行完就回到返回地址处,然后继续执行

我们可以发现每调用以此方法都会有栈帧入栈,当方法运行结束之后,栈帧会弹出 , 同时调用的时候记录返回地址,执行完就回到返回地址处,然后继续执行

 多线程运行的原理

多线程运行原理

public class TestThreadDemo1 {
    public static void main(String[] args) {
        Thread t = new Thread("t1"){
            @Override
            public void run() {
                method(10);//t1线程调用的方法
            }
        };
        t.start();
        method(10);//主线程调用的方法
    }
    public static void method(int x){
        int y = x + 1;
        Object m = method2();
        System.out.println(m);
    }
    public static Object method2(){
        Object n = new Object();
        return n;
    }
}

总结一下就是 :

多线程运行的时候,每一个线程都会有自己的栈帧,线程的栈内存相互独立(每个线程拥有自己的独立的栈内存,栈内存里面有多个栈帧),互不干扰,当系统内没有任何线程执行的时候,整个程序就结束了

什么是线程上下文切换

在多线程运行的时候,CPU过时间片分配算法来循环执行任务 , 任务调度器会给每一个线程分配CPU时间片,当一个线程执行完一个时间片的任务之后就会切换到下一个任务,就会把CPU的使用权交给其他线程,在切换前我们也要保存上一个任务的状态,以便下一次切换回这个任务的时候,可以加载这个任务的状态,继续向下运行,该线程从使用CPU->不使用CPU 就是一次线程的上下文切换(从保存到再加程就是一次上下文切)

发生上下文切换的原因

被动发生上下文切换 

  • 时间片用完

也就是当任务调度器将CPU时间片分配给每一个线程运行,这个时间片总有用完的时候,时间片用完的时候,CPU的使用权就会交给其他线程,这个时候对于当前线程就是发生一次上下文切换.

  • 垃圾回收

垃圾回收会暂停当前工作的线程(就会出现上下文切换),就会让垃圾回收的线程开始回收垃圾

  • 更高优先级的线程要运行

CPU的使用权就要交给优先级更高的线程来使用,当前线就会暂停(就是一次线程的上下文切换),当这个优先级更高的线程运行完了之后,由任务调度器会重新为当前线程分配CPU时间片

主动发生上下文切换

  • 线程自己调用了sleep,yield,wait,join,park,synchronized,lock等方法

当完成一次线程的上下文切换的时候,操作系统就要保存当前线程的状态(包括需要使用程序计数器记住下一条JVM的指令执行地址,还有一些状态信息,虚拟机栈的栈帧信息,如局部变量,操作数栈,返回地址....),并且恢复另一个线程的状态

为啥要保存当前线程的状态呢 ?

原因是当我们下次在运行的时候,CPU就知道要从哪开始运行了.

使用程序计数器就可以保存当前代码执行到哪里了(也就 是记住下一跳JVM指令的地址).

频繁的上下文切换会影响性能.

并不是线程数越多越好,当CPU核心数 < 线程数,线程的上下文切换的成本就会提高.频繁的上下文切换就会影响性能.

线程数越多越好么 ?

不是的,当线程数超过CPU核心数的时候,CPU就要在这么多线程轮流切换, 线程的上下文切换成本会提高,频繁的上下文切换影响程序的性能.

线程方法

start与run

start与run的区别

  • 调用start方法才是真正的启动一个新的线程执行,调用run方法只是执行了一个普通方法不没有真正的创建线程让线程执行,run方法只是封装了要被线程执行的任务
  • 调用start方法不会立即执行任务 , 调用run就可以立即执行. 

因为调用start方法的时候不是立即执行线程任务,而是从新建状态变为就绪状态,就绪状态就是已经获取到了除CPU资源以外的其他资源,等待任务调度器分配给当前线程CPU时间片,才会真正的执行任务. 所以调用start方法不会立即执行任务. 但直接调用run,run相当于是普通方法,可以立即执行

  • 重复调用start方法就会抛出异常,对于run方法为普通方法可以重复调用

为什么调用start()方法时会执行run()方法,而不直接执行run()方法?

首先我们使用多线程的原因就是提高速度,创建多个线程来执行任务.

  • 调用start方法才会启动一个新的线程,并在新线程中执行run方法的代码.因为start方法会做一些准备工作,包括线程分配资源,初始化线程上下文,然后等到获取到CPU时间片, 调用操作系统相关API创建新的线程
  • 直接调用run方法,会在当前线程中执行run方法代码,并不会启动一个新的线程去执行任务

为什么start方法不能重复调用?而run方法却可以?

当我们重复调用start方法的时候就会出现异常...

可以看源码进行分析 :

0->表示NEW状态.

调用start方法的时候,首先会先判断线程状态(threadStatus)是不是等于0(是不是新建状态),如果不是新建状态了,那就会抛出异常.原因是当调用第一次start方法的时候,线程状态从NEW变为了Runnable状态,由于线程状态是不可逆的(也就是说不能在从Runnable->NEW了),所以JVM在判断当前线程状态已经不是NEW状态了,他就会抛出异常.

即 : 重复调用 start()会抛出异常

可以直接调用 Thread 类的 run 方法吗?

可以通过Thread类对象直接调用run方法,run方法就是一个普通方法,用于封装线程执行的任务, 如果直接调用run方法会直接在当前线程(一般为主线程)执行,不会新启动一个线程去执行任务.

sleep,yield,wait

sleep与yield的区别

  • sleep()方法使线程进入阻塞状态,暂停执行指定的时间,在时间结束后重新进入就绪状态. yield()方法使线程放弃当前获得的CPU时间片,重新回到就绪状态
  • sleep()方法给其他线程运行机会时不考虑线程的优先级,因此会给低优先级的线程以运行的机会,yield()方法只会给相同或更高优先级的线程以运行的机会;
  • 线程执行sleep()方法后进入阻塞(blocked)状态,而执行yield()方法后进入就绪(ready)状态;
  • sleep()方法声明抛出InterruptedException,而yield()方法没有声明任何异常;

sleep与wait的区别

共同点:两者都可以暂停线程的执行.

两者的区别 :

  • wait() 通常被用于线程间交互/通信,sleep()通常被用于暂停执行。
  • sleep 不需要强制与 synchronized 配合使用, 但是 wait 必须和 synchronized 一起使用.调用wait方法前必须先获取wait对象的锁 
  • 调用sleep方法后不会释放对象锁, 但调用wait方法之后就会释放对象锁,允许其他线程获取锁
  • wait() 方法被调用后,线程不会自动苏醒,需要别的线程调用同一个对象上的 notify()或者 notifyAll() 方法.或者也可以使用 wait(long timeout) 超时后线程会自动苏醒。sleep()方法执行完成后,线程会自动苏醒 并且他们都可以被打断唤醒
  • sleep()方法调用后, 线程进入TIMED_WAITING状态, wait()方法调用后未被唤醒前是WATING状态。 wait(n),sleep(n)两者共同点就是 : 线程的状态都为TIMEDWAITING
  • sleep是Thread类的静态方法,而wait是所有对象都有的方法,是Object的方法

为什么 wait() 方法不定义在 Thread 中?

因为调用wait方法前必须获取对象锁,让获取对象锁的线程等待,然后就会自动释放当前线程占有的对象锁.

每一个对象都拥有对象锁,要想让获取到对象锁的线程进入WAITING状态,操作的是对应的对象,而不是线程.

为什么 sleep() 方法定义在 Thread 中?

sleep方法是让当前线程暂停执行,没有涉及到对象类,就不需要获取对象锁.

线程的优先级

线程优先级会提示( hint )调度器优先调度该线程,但它仅仅是一个提示,调度器可以忽略它  
如果 cpu 比较忙,那么优先级高的线程会获得更多的时间片,但 cpu 闲时,优先级几乎没作用

Thread.sleep(0)的作用是什么?

Thread的sleep会让线程暂时释放CPU资源,然后进入TIMED_WAITING状态,等待指定时间之后在尝试获取CPU时间片.

Thread.sleep(0),就是让当前线程释放一下CPU时间片,然后重新开始争抢.

可以用来做线程调度,比如某个线程长时间占用CPU资源,这个时候sleep(0)让线程主动释放CPU时间片,让其他线程可以进行一次公平的竞争

join方法

  • join()方法的作用是什么?

join方法就是用于一个线程等待另外一个线程执行完毕,然后自己可以利用那个线程执行完的结果做一些其他事情.

  • join()方法与sleep()方法有什么区别?
  1. join方法用于一个线程等待另外一个线程执行完毕, sleep方法用于让当前线程暂停一段时间
  2. join方法必须在线程对象上调用,而 sleep() 是Thread类的静态方法,通过Thread类调用
  3. join方法进入WAITING状态, sleep方法进入TIMED_WAITING状态
  • 多个线程之间如何正确使用 join() 方法来协同工作?/如何使用 join() 方法来保证某些线程在其它线程执行完毕后再执行?

多个线程之间使用join方法,主要是确保线程之前的顺序执行.

我们需要创建多个协作的线程,哪个线程需要等待另外一个线程结束,就在该线程内部调用另外一个线程的join方法

比如 main线程等待t1,t2结束,那么就在主线程调用t1.join,t2.join

public static void main(String[] args) throws InterruptedException {
    Thread t1 = new Thread(() -> {
        System.out.println("t1");
    }, "t1");
    Thread t2 = new Thread(() -> {
        System.out.println("t2");
    }, "t2");
    t1.start();
    t2.start();
    t1.join();
    t2.join();
    System.out.println("main");
}

如果t1要想等待t2结束,那么就在t1线程中调用t2.join方法

public static void main(String[] args) throws InterruptedException {
    Thread t2 = new Thread(() -> {
        System.out.println("t2");
    }, "t2");
    Thread t1 = new Thread(() -> {
        try {
            t2.join();
        } catch (InterruptedException e) {
            throw new RuntimeException(e);
        }
        System.out.println("t1");
    }, "t1");
    t1.start();
    t2.start();
    t1.join();
    System.out.println("main");
}
  • 如果一个线程已经完成了它的任务,再次调用它的 join() 方法会发生什么?

如果一个线程已经完成了它的任务,再次调用它的 join() 方法将不会发生任何变化,因为已经没有需要等待的内容了。

interrupt方法

如何停止一个正在运行的线程? /如何优雅的关闭线程

有三种方式可以用来停止线程

  • 使用标志位,来终止其他线程的run方法
@Slf4j(topic = "c.TestInterrupt")
public class InterruptTest1 {
    //1.使用自带标记位
    private static volatile boolean flg = false;
    public static void main(String[] args) throws InterruptedException {
        new Thread(()->{
            while (!flg) {
                log.debug("test...");
                try {
                    Thread.sleep(1000);
                } catch (InterruptedException e) {
                    throw new RuntimeException(e);
                }
            }
        },"t1").start();
        Thread.sleep(2000);
        flg = true;
    }
}
  • 使用stop方法强行终止
  • @Slf4j(topic = "c.InterruptTest2")
    public class InterruptTest2 {
        public static void main(String[] args) throws InterruptedException {
            Thread t1 = new Thread(() -> {
                while(true) {
                    log.debug("test...");
                    try {
                        Thread.sleep(1000);
                    } catch (InterruptedException e) {
                        throw new RuntimeException(e);
                    }
                }
            }, "t1");
            t1.start();
            Thread.sleep(2000);
            t1.stop();
        }
    }
  • 使用interrupt方法打断该线程
  1. 打断正在阻塞的线程.(比如调用了sleep,wait,join方法,都会进入阻塞状态这些方法都要抛出InterrupedException)
public static void main(String[] args) throws InterruptedException {
    //1.打断阻塞的线程
    Thread t1 = new Thread(() -> {
        log.debug("sleep...");
        try {
            Thread.sleep(1000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }, "t1");
    t1.start();
    Thread.sleep(600);
    t1.interrupt();
    log.debug(t1.isInterrupted()+"");
}

被阻塞的线程打断标记会被设置为false,将来打断标记可以用来判断这个线程是继续运行还是就此终止.

  1. 打断正常运行的线程.
@Slf4j(topic = "c.InterruptTest4")
public class InterruptTest4 {
    public static void main(String[] args) throws InterruptedException {
        //1.打断正在运行的线程
        Thread t1 = new Thread(() -> {
            while(true) {
                log.debug("sleep");
                boolean interrupted = Thread.currentThread().isInterrupted();
                if(interrupted) {
                    //由被打断线程自己去决定,是继续运行还是停止
                   break;
                }
            }
        }, "t1");
        t1.start();
        t1.interrupt();
    }
}

打断正在运行的线程也是使用t.interrupt()方法来进行打断, 会把打断标记设置为true,被打断线程可以根据打断标记来判断,接下来是继续运行还是就此停止.

错误停止线程的方式

  • 使用stop方法强行终止
@Slf4j(topic = "c.InterruptTest2")
public class InterruptTest2 {
    public static void main(String[] args) throws InterruptedException {
        Thread t1 = new Thread(() -> {
            while(true) {
                log.debug("test...");
                try {
                    Thread.sleep(1000);
                } catch (InterruptedException e) {
                    throw new RuntimeException(e);
                }
            }
        }, "t1");
        t1.start();
        Thread.sleep(2000);
        t1.stop();
    }
}

不建议使用stop方法来终止线程,这样不是优雅的终止,而是强制终止.

interrupt, isInterrupted, interrupted的区别是什么

  • void interrupt() :interrupt() 方法只是给当前线程设置了一个打断标记,把打断标记置为true,至于后续是打断还是继续运行,由当前线程自己决定(料理完后事在退出).对于打断阻塞的线程(调用了wait方法,sleep方法,join方法),会认为是异常终止,会把打断标记置为false
  • boolean isInterrupted() 方法: 检测当前线程是否被中断。
  • boolean interrupted() 方法:检测当前线程是否被中断, 与isInterrupted 不同的是,该方法如果发现当前线程被中断, 则会清除中断标志.

notify和notifyAll有什么区别

  • notify和notifyAll都是操作对象的,所以都被定义在Object中,都是Object的方法
  • notify 是把在等待队列(WaitSet)中等待的线程,随机唤醒一个
  • notifyAll : 是把在等待队列(WaitSet)中等待的线程全部唤醒

还有一点要注意的是,从等待队列中被唤醒并不能立即获取到CPU的使用权,只是表示他们可以竞争锁了,竞争到锁之后才有机会被CPU调度

notifyAll虽然唤醒的是waitset中的所有线程,但是最终只能有一个线程获取/竞争到锁.

线程的状态

线程有5种状态和6种状态两种说法

线程的5种状态是指操作系统层面上的 , 线程的6种状态是指JavaAPI层面上的

线程的5种状态

线程的5种状态是指操作系统层面上的

  • 初始状态 : 仅仅是语言层面创建了线程对象,还没有与操作系统相关联.比如new 了一个Thread对象还没有调用start方法
  • 可运行状态 : 也可以理解为就绪状态,该线程已经被创建了,与操作系统相关联,可以由CPU调度,但是线程调度器还没有给该线程分配CPU时间片
  • 运行状态 : 给该线程分配了CPU时间片,线程处于正在运行的状态
    • 这里运行状态和可运行状态是能进行上下文切换的,当CPU时间片用完了,线程就会从运行状态变为可运行状态.,导致线程上下文切换
  • 阻塞状态 : 该线程调用了一些阻塞API,如BIO读写文件,这时该线程实际不会用到CPU,就会导致上下文切换,从运行状态变为阻塞状态.线程调度器不会分配给[阻塞状态线程]CPU时间片的,但是会分配给可运行状态的线程. 当读写文件操作完毕,会由操作系统来唤醒阻塞状态的线程, 就会从阻塞状态变为可运行状态
  • 终止状态 : 当当前线程的所有代码执行完成之后,表示线程已经执行完毕,生命周期已经结束了,不会再转换为其他状态.

线程的六种状态

线程的6种状态是JavaAPI层面上的.

根据Thread.State 枚举 ,分为6种状态.

  • NEW : 仅仅是语言层面创建了线程对象,还没有与操作系统相关联.比如new 了一个Thread对象还没有调用start方法
  • RUNNABLE : JavaAPI层面上状态涵盖了操作系统层面的[可运行状态],[运行状态] 和[阻塞状态]
  • 阻塞状态在Java里面细分为 BLOKED,WAITING,TIMED_WAITING
  • BLOKED : 当t线程调用synchronized(obj) 获取对象锁时竞争失败,会进入BLOCKED状态
  • WAITING : 当调用线程的join方法,LockSupport.park()方法或者用synchronized(obj)获取了对象锁后,调用obj.wait()方法时.
  • TIMED_WAITING : 当用synchronized获取对象锁后,调用wait(long n)方法时候,当调用join(long n) ,调用sleep(long n),或者调用LockSupport的有时限的方法
  • TERMINATED : 当当前线程的所有代码执行完成之后,表示线程已经执行完毕,生命周期已经结束了,不会再转换为其他状态.

线程状态之间的变化

  • NEW --> RUNNABLE

当调用了 t.start()方法的时候,由NEW --> RUNNABLE

  • RUNNABLE <--> WAITING

调用wait方法,join方法,LockSupport.park() 线程就从 RUNNABLE --> WAITING

  1. 对于join方法和park方法 当其他线程调用interrupt方法打断该线程就会 从 WAITING-->RUNNABLE状态
  2. 对于wait方法
  • 调用了 obj.notify() ,obj.notifyAll() ,t.interrupt() 时候
    • 竞争锁成功,t线程从 WAITING --> RUNNABLE
    • 竞争锁失败,t线程从 WAITING --> BLOCKED

比如两个线程竞争同一把锁,其中主线程使用notifyAll把它们全唤醒,那必然有一个线程竞争锁成功从WAITING --> RUNNABLE,必然有一个线程竞争锁失败从WAITING --> BLOCKED

  • RUNNABLE <--> TIMED_WAITING
  1. 当前线程调用 t.join(long n) 方法, Thread.sleep(long n) , LockSupport.parkNanos(long nanos) obj.wait(long n)方法 ,就会进入 TIMED_WAITING状态

     2. 对于t.join(long n) 方法, Thread.sleep(long n) , LockSupport.parkNanos(long nanos) 当到达指定时间就会从 TIMED_WAITING --> RUNNABLE

        对于 obj.wait(long n)方法

  • 当t线程等待时间超过了n毫秒,或调用obj.notify(),obj.notifyAll(),t.interrupt() 时
    • 竞争锁成功,t 线程从 TIMED_WAITING --> RUNNABLE
    • 竞争锁失败,t 线程从 TIMED_WAITING --> BLOCKED
  • RUNNABLE <--> BLOCKED
  • 当t线程用synchronized(obj)获取了对象锁时如果竞争失败就会从 RUNNABLE-->BLOCKED
  • 持有obj锁线程的同步代码块执行完毕,会唤醒该对向上的所有BLOKED的线程,如果其中t线程竞争成功,从BLOKED-->RUNNABLE ,其他失败的线程仍然是 BLOCKED.

RUNNABLE --> TERMINATED

当当前线程的所有代码执行完成之后,表示线程已经执行完毕,生命周期已经结束了,不会再转换为其他状态.

线程死亡的几种情况

  1. 线程正常执行完毕:当线程执行完自己的任务后,线程会自然结束并死亡。

  2. 异常终止:如果线程中抛出未捕获的异常并且没有对该异常进行处理,线程将会异常终止并死亡。这种情况下,可以通过设置异常处理程序来捕获并处理线程中的异常,避免线程死亡。

  3. 调用stop()方法:stop()方法是一个被废弃的方法,它会强制终止一个线程的执行。这种方式容易引发线程安全问题和资源泄漏,不推荐使用。

线程阻塞的几种情况

  • 阻塞状态在Java里面细分为 BLOKED,WAITING,TIMED_WAITING
  • BLOKED : 当t线程调用synchronized(obj) 获取对象锁时竞争失败,会进入BLOCKED状态
  • WAITING : 当调用线程的join方法,LockSupport.park()方法或者用synchronized(obj)获取了对象锁后,调用obj.wait()方法时.
  • TIMED_WAITING : 当用synchronized获取对象锁后,调用wait(long n)方法时候,当调用join(long n) ,调用sleep(long n),或者调用LockSupport的有时限的方法

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

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

相关文章

自学黑客(网络安全),一般人我劝你还是算了吧(自学网络安全学习路线--第十九章 口令破解与防御技术下)【建议收藏】

文章目录 一、自学网络安全学习的误区和陷阱二、学习网络安全的一些前期准备三、自学网络安全学习路线一、口令攻击的综合应用1、Windows NT, 2000口令攻击2、Windows XP, 2003口令攻击3、Unix系统口令攻击4、远程口令攻击 二、口令攻击的防御1、口令攻击防御概述2、保持口令的…

RS485转Profinet通讯

RS485转Profinet通讯 概述系统组成流量积算仪网关 软件总结 概述 一个支持RS485的流量积算仪的数据要被Profinet的PLC读取。制作一个网关&#xff0c;实现RS485到Profinet的转换。 系统组成 流量积算仪 支持RS485通讯&#xff0c;通讯协议是modbus RTU。采用功能码3可以读取…

非GUI模式下如何传参

非GUI模式下如何传参 Jmeter 有两种模式&#xff0c;GUI模式和非GUI模式&#xff0c;通常使用GUI模式编辑脚本&#xff0c;使用非GUI模式运行压测&#xff0c;官网上有强调&#xff0c;尽量使用非GUI模式&#xff0c;因为GUI模式下&#xff0c;jmeter UI组件本身在压测过程中会…

论文浅尝 | SimKGC:基于预训练语言模型的简单对比知识图谱补全

笔记整理&#xff1a;李雅新&#xff0c;天津大学硕士&#xff0c;研究方向为知识图谱补全 链接&#xff1a;https://dl.acm.org/doi/10.1145/3539597.3570483 动机 知识图谱补全 (KGC) 旨在对已知事实进行推理并推断缺失的链接。基于文本的方法从自然语言描述中学习实体表示&a…

Spring Boot 中的 @EnableConfigurationProperties 注解

Spring Boot 中的 EnableConfigurationProperties 注解 在 Spring Boot 中&#xff0c;EnableConfigurationProperties 注解是一个非常有用的注解&#xff0c;它可以用于启用对特定配置类的支持。在本文中&#xff0c;我们将深入探讨 EnableConfigurationProperties 注解&…

ST CubeMX 实现6对PWM同步输出/互补输出/三相PWM输出

频率为1KHz的6对PWM波形 原理:定时器1为主模式,定时器8为从模式,TIM1的定时器使能操作作为触发输出[TRGO]触发TIM8并使能TIM8的计数器,同时输出两路频率、占空比以及脉冲数量(小于256个,高级定时器重复计数功能为8位)可调PWM波形。 Tim1 参数配置 Tim8参数配置 未同步输出…

CC2530 GPIO口输出配置说明

第一章 原理图分析 CC2530核心板上带有两颗晶振:第一颗频率为32MHZ,第二颗频率为32.768KHZ CC250正常运行的时候,需要一个高频的时钟信号和一个低频的时钟信号。 高频时钟信号,主要供给CPU,保证程序的运行。 低频时钟信号,主要供给看门狗、睡眠定时器等片上外设。 按…

【强化学习】常用算法之一 “TRPO”

作者主页&#xff1a;爱笑的男孩。的博客_CSDN博客-深度学习,活动,python领域博主爱笑的男孩。擅长深度学习,活动,python,等方面的知识,爱笑的男孩。关注算法,python,计算机视觉,图像处理,深度学习,pytorch,神经网络,opencv领域.https://blog.csdn.net/Code_and516?typeblog个…

StorageGRID——开放式的 S3 对象存储,可大规模管理非结构化数据

StorageGRID——开放式的 S3 对象存储&#xff0c;可大规模管理您的非结构化数据 专为混合多云体验打造的对象存储 StorageGRID 通过简化的平台为对象数据提供更强大的数据管理智能。由于 StorageGRID 利用 S3&#xff0c;因此可以轻松地连接混合云工作流&#xff0c;提供流畅…

C++ - 20230628

一. 思维导图 二. 练习 1) 总结类和结构体的区别 本身的访问级别不同struct是值类型&#xff0c;class是引用类型struct在栈&#xff0c;适合处理小型数据。class在堆区&#xff0c;适合处理大型逻辑和数据。 2) 定义一个矩形类&#xff08;Rectangle&#xff09;&#xff…

基于Java+SpringBoot+vue的高校学生党员发展管理系统设计与实现

博主介绍&#xff1a;✌擅长Java、微信小程序、Python、Android等&#xff0c;专注于Java技术领域和毕业项目实战✌ &#x1f345;文末获取源码联系&#x1f345; &#x1f447;&#x1f3fb; 精彩专栏推荐订阅&#x1f447;&#x1f3fb; 不然下次找不到哟 Java项目精品实战案…

我是怎么把win11一步一步变成Mac的

目录 【三指拖动】 【空格预览】 【切换Ctrl和Alt】 【使用Linux命令】 【其它】 之前很长一段时间在MacBook上面开发习惯了&#xff0c;然后因为一些原因现在换到了windows上面&#xff0c;不管是使用上还是系统上都很不习惯&#xff0c;因此做了一些改造&#xff0c;…

LTD232次升级 | 社区新增PC版首页 • 名片新增卡片样式、可展示传真 • 导航数据可官微中心管理 • 个人中心可定制

1、社区支持PC版首页 2、名片小程序新增一种全局卡片样式、支持显示传真 3、官微中心新增导航管理 4、手机版商城个人中心支持版块配置 5、新增一组新闻轮播模块 01 用户社区应用 1) 新增PC版社区首页 在本次升级中&#xff0c;我们为用户社区应用新增了PC版的首页。 开…

【探索 Kubernetes|作业管理篇 系列 15】DaemonSet 的”过人之处“

前言 大家好&#xff0c;我是秋意零。 在上一篇中&#xff0c;我们讲解了 StatefulSet 的存储状态&#xff1b;我们发现&#xff0c;它的存储状态&#xff0c;就是利用了 PV 与 PVC 的设计。StatefulSet 自动为我们创建 PVC 并且以 <pvc-name>-<pod-name>-<编…

selenium模拟!看这篇就够了

介绍 Selenium是一个用于自动化Web浏览器测试的开源工具&#xff0c;它支持多种Web浏览器&#xff08;如Google Chrome、Firefox、Safari等&#xff09;和操作系统&#xff08;如Windows、Mac和Linux&#xff09;。Selenium可以模拟用户在Web浏览器中的行为&#xff0c;例如点…

ssm汉语言学习应用系统APP -计算机毕设 附源码80400

ssm汉语言学习应用系统APP 摘 要 在信息飞速发展的今天&#xff0c;网络已成为人们重要的信息交流平台。每天都有大量的农产品需要通过网络发布&#xff0c;为此&#xff0c;本人开发了一个基于Android模式的汉语言学习应用系统。 对于本汉语言学习应用系统的设计来说&#x…

十、云尚办公系统-员工端审批

云尚办公系统&#xff1a;员工端审批 B站直达【为尚硅谷点赞】: https://www.bilibili.com/video/BV1Ya411S7aT 本博文以课程相关为主发布&#xff0c;并且融入了自己的一些看法以及对学习过程中遇见的问题给出相关的解决方法。一起学习一起进步&#xff01;&#xff01;&…

回收站删除的文件怎么恢复?4招快速搞定!

求救求救&#xff01;我刚刚一个不小心就把回收站清空了&#xff01;但是我回收站里还有需要恢复的文件&#xff0c;这次一不小心清空了回收站&#xff0c;我的重要文件还有机会找回来吗&#xff1f;希望大家帮帮我! 对于部分朋友来说&#xff0c;回收站可能不仅仅是一个垃圾文…

Selenium 不开启浏览器页面执行测试用例

实际工作中会遇到不开启浏览器页面来执行测试用例的情况&#xff0c;可以通过ChromeOptions来实现 ChromeOptions是chromedriver支持的浏览器启动选项 Google 针对 Chrome 浏览器 59版 新增加的Chrome-headless 模式&#xff0c;可以在不打开UI界面的情况下使用 Chrome 浏览器…

【Java高级编程】多线程

多线程 1、基本概念&#xff1a;程序、进程、线程1.1、程序1.2、进程1.3、线程1.4、单核CPU和多核CPU的理解1.5、并行与并发1.6、使用多线程的优点1.7、何时需要多线程 2、线程的创建和使用2.1、创建多线程的方式一&#xff1a;继承Thread类2.2、Thread类的有关方法2.3、线程的…