多线程(创建多线程的五种方式,线程状态, 线程中断)

简介: 多线程(创建多线程的五种方式,线程状态, 线程中断)

创建多线程

创建多线程有多种形式, 此处介绍五种


继承 Thread 类

class MyThread extends Thread {
    @Override
    public void run() {
        System.out.println("hello world!1");
    }
}

public class Main{
    public static void main(String[] args) {
        // 继承 Thread 方法
        MyThread t = new MyThread();
        t.start();
       
    }
}

实现 Runnable 接口

class MyRunnable implements Runnable {
    @Override
    public void run() {
        System.out.println("hello runnable!");
    }
}

public class Main{
    public static void main(String[] args) {
        // 实现 Runnable 接口
        Thread t = new Thread(new MyRunnable());
        t.start();
    }
}

匿名内部类, 继承 Thread 类

public class Main{
    public static void main(String[] args) {
         // 匿名内部类, 继承 Thread 类
        // 1. 创建了一个 Thread 的子类 (子类没有名字, 因此叫 "匿名")
        // 2. 创建了子类的实例, 并且让 t 指向该实例
        Thread t = new Thread() {
            @Override
            public void run() {
                System.out.println("hello ");
            }
        };
        t.start();
    }
}

匿名内部类实现 Runnable 接口

public class Main{
    public static void main(String[] args) {
        // 匿名内部类, 实现 Runnable 接口
        // 1. 创建了一个类实现 Runnable
        // 2. 创建了类的实例, 并且传给 Thread 的构造方法
        Thread t = new Thread(new Runnable() {
            @Override
            public void run() {
                System.out.println("hello 匿名内部类!");
            }
        });
        t.start();
    }
}

lambda 表达式

public class Main{
    public static void main(String[] args) {
       // lambda 表达式
        Thread t = new Thread(() -> {
            System.out.println("hello world!");
        });
        t.start();
    }
}

线程的状态

线程是CPU调度的基本单位, 所以这里的状态, 也是指 Java线程的状态, 可以使用 Thread.getState() 方法可以获取线程的状态

public class ThreadDemo5 {
    public static void main(String[] args) throws InterruptedException {
        Thread t1 = new Thread(() -> {
            for (int i = 0; i < 10000; i++) ; //该行代码主要是让子线程跑一会儿, 以便主线程获取到子线程的 RUNNABLE 方法
        });
        System.out.println(t1.getState()); //NEW
        t1.start();
        System.out.println(t1.getState()); //RUNNABLE

        Thread.sleep(3000);
        System.out.println(t1.getState()); //TERMINATED

    }
}

NEW: 创建了 Thread 对象, 但是还没调用 start() 方法 (内核中还没创建对应的 PCB)

TERMINATED: 内核中PCB已经执行完毕了, 但是 Thread 对象还在 (执行过 start(), 并且 run() 方法执行完毕)

RUNNABLE: 可运行的 (包含两类线程)

  • 正在CPU上运行的线程
  • 在就绪队列中, 随时可以去CPU上运行的线程

WAITING: wait / join /IO … 产生的阻塞

TIMED_WAITING: sleep() 产生的阻塞

BLOCKED: 等待锁产生的阻塞


线程的状态转换



线程中断

线程中断不是直接终止中断, 而是通知线程, 你应该要中止了, 具体线程中止不中止, 还是看代码里面的具体逻辑

下面介绍两种线程中断的方式 (其实本质上是一种, 都是通过标志位来通知线程中断)


线程中断的实现 – 自写一个标志位控制线程中断

// 使用标志位来控制线程中断 (只是通知, 不是强制中断)
public class Main{
    public static boolean f = true; //自写的标志位

    public static void main(String[] args) throws InterruptedException {
        Thread t = new Thread(() -> {
            int n = 0;
            while(f) { //主线程修改标志位, 子线程就能读取到, 以此来判断子线程是否要中断
                System.out.println("hello " + n);
                try {
                    Thread.sleep(500);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                n++;
            }
        });

        t.start();
        Thread.sleep(3000);
        System.out.println(t.getState());
        f = false;
        System.out.println(t.getState());
    }
}

线程中断的实现 – 通过 Thread 自带的标志位来进行线程中断

// 使用 Thread 自带的标志位来控制线程中断 interrupted
public class ThreadDemo3 {
    public static void Main(String[] args) throws InterruptedException {
        Thread t = new Thread(() -> {
            int x = 0;
            while(!Thread.currentThread().isInterrupted()) {
        // Thread.currentThread() -> 来获取当前线程
        // Thread.currentThread().isInterrupted() -> 判断当前线程是否被中断, 被中断返回true
        // 那 !Thread.currentThread().isInterrupted() ->  就是如果线程被中断, 返回 false, 即退出 while 循环

                try {
                    System.out.println("hello " + x++); //懒人写法,测试用的, 正经写代码别这么写(年终奖警告!!)
                    Thread.sleep(500);
                } catch (InterruptedException e) {
                    // interrupt会触发 sleep 内部的异常, 因此可以在这里面进行线程被 interrupted 之后的事情
                    e.printStackTrace();
                    System.out.println("中断执行, sleep 被唤醒, 进入异常中 ... ");  
                    break;
                }
            }
        });

        t.start();
        Thread.sleep(3000);
        t.interrupt();
        
        // interrupted 做的事
        // 1.将线程内部的标志位置 true
        // 2.在 sleep 过程中被调用的话, interrupt会触发 sleep 内部的异常, 导致 sleep 提前退出

        // sleep 唤醒后做的事
        // 1.执行异常处理
        // 2.sleep唤醒后会清空标志位 (置 false)
}

下面是上述代码的重点, 单拎出来以示尊重~~

interrupt() 做的事:

1.将线程内部的标志位置 true

2.在 sleep 过程中被调用的话, interrupt会触发 sleep 内部的异常, 导致 sleep 提前退出

sleep 唤醒后做的事:

1.执行异常处理

2.sleep唤醒后会清空标志位 (置 false)

目录
相关文章
|
4天前
|
数据采集 负载均衡 安全
LeetCode刷题 多线程编程九则 | 1188. 设计有限阻塞队列 1242. 多线程网页爬虫 1279. 红绿灯路口
本文提供了多个多线程编程问题的解决方案,包括设计有限阻塞队列、多线程网页爬虫、红绿灯路口等,每个问题都给出了至少一种实现方法,涵盖了互斥锁、条件变量、信号量等线程同步机制的使用。
LeetCode刷题 多线程编程九则 | 1188. 设计有限阻塞队列 1242. 多线程网页爬虫 1279. 红绿灯路口
|
12天前
|
Java Spring
spring多线程实现+合理设置最大线程数和核心线程数
本文介绍了手动设置线程池时的最大线程数和核心线程数配置方法,建议根据CPU核数及程序类型(CPU密集型或IO密集型)来合理设定。对于IO密集型,核心线程数设为CPU核数的两倍;CPU密集型则设为CPU核数加一。此外,还讨论了`maxPoolSize`、`keepAliveTime`、`allowCoreThreadTimeout`和`queueCapacity`等参数的设置策略,以确保线程池高效稳定运行。
72 10
spring多线程实现+合理设置最大线程数和核心线程数
|
20天前
|
Java 数据库 Android开发
一个Android App最少有几个线程?实现多线程的方式有哪些?
本文介绍了Android多线程编程的重要性及其实现方法,涵盖了基本概念、常见线程类型(如主线程、工作线程)以及多种多线程实现方式(如`Thread`、`HandlerThread`、`Executors`、Kotlin协程等)。通过合理的多线程管理,可大幅提升应用性能和用户体验。
35 15
一个Android App最少有几个线程?实现多线程的方式有哪些?
|
6天前
|
Python
5-5|python开启多线程入口必须在main,从python线程(而不是main线程)启动pyQt线程有什么坏处?...
5-5|python开启多线程入口必须在main,从python线程(而不是main线程)启动pyQt线程有什么坏处?...
|
22天前
|
Java 数据库 Android开发
一个Android App最少有几个线程?实现多线程的方式有哪些?
本文介绍了Android应用开发中的多线程编程,涵盖基本概念、常见实现方式及最佳实践。主要内容包括主线程与工作线程的作用、多线程的多种实现方法(如 `Thread`、`HandlerThread`、`Executors` 和 Kotlin 协程),以及如何避免内存泄漏和合理使用线程池。通过有效的多线程管理,可以显著提升应用性能和用户体验。
38 10
|
3天前
|
NoSQL 网络协议 Unix
1)Redis 属于单线程还是多线程?不同版本之间有什么区别?
1)Redis 属于单线程还是多线程?不同版本之间有什么区别?
8 0
|
4天前
|
Java
COMATE插件实现使用线程池高级并发模型简化多线程编程
本文介绍了COMATE插件的使用,该插件通过线程池实现高级并发模型,简化了多线程编程的过程,并提供了生成结果和代码参考。
|
29天前
|
存储 Ubuntu Linux
C语言 多线程编程(1) 初识线程和条件变量
本文档详细介绍了多线程的概念、相关命令及线程的操作方法。首先解释了线程的定义及其与进程的关系,接着对比了线程与进程的区别。随后介绍了如何在 Linux 系统中使用 `pidstat`、`top` 和 `ps` 命令查看线程信息。文档还探讨了多进程和多线程模式各自的优缺点及适用场景,并详细讲解了如何使用 POSIX 线程库创建、退出、等待和取消线程。此外,还介绍了线程分离的概念和方法,并提供了多个示例代码帮助理解。最后,深入探讨了线程间的通讯机制、互斥锁和条件变量的使用,通过具体示例展示了如何实现生产者与消费者的同步模型。
|
2月前
|
监控 安全 Java
Java多线程调试技巧:如何定位和解决线程安全问题
Java多线程调试技巧:如何定位和解决线程安全问题
82 2