【Java】了解线程 Thread 类的使用,如何创建、终止、等待一个线程以及获取线程的状态

线程是什么

线程是操作系统中调度的基本单位,是比进程更小的执行单元。线程在进程内部运行,共享该进程的资源,如内存和文件句柄,但每个线程都有自己的执行栈和程序计数器。

线程的主要特点包括:

  1. 轻量级:线程相较于进程更加轻量,创建和销毁的开销较小。
  2. 共享资源:同一进程中的线程共享该进程的内存空间和资源,从而可以更高效地进行数据交换。
  3. 并发执行:多个线程可以并发执行,充分利用多核处理器,提高程序的执行效率。
  4. 简化管理:线程的切换和管理相对于进程更为简单和迅速,有助于提升系统的响应速度。

线程的使用在现代操作系统中非常普遍,尤其是在需要高并发和高性能的应用场景中,例如网络服务器和多任务应用程序等。

为什么要有线程

首先, "并发编程" 成为 "刚需"。

单核 CPU 的发展遇到了瓶颈. 要想提高算力, 就需要多核 CPU. 而并发编程能更充分利用多核 CPU资源。
有些任务场景需要 "等待 IO", 为了让等待 IO 的时间能够去做一些其他的工作, 也需要用到并发编
程。

其次, 虽然多进程也能实现 并发编程, 但是线程比进程更轻量。

  • 创建线程比创建进程更快.
  • 销毁线程比销毁进程更快.
  • 调度线程比调度进程更快.

创建出一个线程

在Java中,可以通过两种主要方式创建线程:继承Thread类和实现Runnable接口。下面分别介绍这两种方式,并附上代码示例。

方法一:继承 Thread 类

  1. 创建一个子类,继承Thread类,并重写run()方法,该方法包含了线程的执行代码。
  2. 创建子类的实例,然后调用start()方法来启动线程。

示例代码:

class MyThread extends Thread {@Overridepublic void run() {// 线程执行的代码for (int i = 0; i < 5; i++) {System.out.println("Thread running: " + i);try {Thread.sleep(1000); // 暂停1秒} catch (InterruptedException e) {e.printStackTrace();}}}
}public class ThreadExample {public static void main(String[] args) {Thread thread = new MyThread(); // 创建线程对象thread.start(); // 启动线程}
}

方法二:实现 Runnable 接口

  1. 创建一个类,实现Runnable接口,并实现run()方法。
  2. 创建Runnable接口的实例,将其传递给Thread构造函数,然后调用start()方法启动线程。

示例代码:

class MyRunnable implements Runnable {@Overridepublic void run() {// 线程执行的代码for (int i = 0; i < 5; i++) {System.out.println("Runnable thread running: " + i);try {Thread.sleep(1000); // 暂停1秒} catch (InterruptedException e) {e.printStackTrace();}}}
}public class RunnableExample {public static void main(String[] args) {MyRunnable myRunnable = new MyRunnable(); // 创建Runnable实例Thread thread = new Thread(myRunnable); // 将Runnable实例传递给Threadthread.start(); // 启动线程}
}

解析:

  • 在这两个示例中,我们创建了一个简单的线程,该线程在运行时每秒打印一次数字(0到4)。
  • 使用Thread.sleep(1000)使线程暂停1秒,这样可以模拟一些耗时的操作,也使得输出不至于淹没在快速的执行中。
  • 调用start()方法时,Java虚拟机会调用线程的run()方法,而不是直接调用run()。这保证了线程的正确启动和管理。

总结:

通过这两种方式,Java允许灵活地创建和管理线程,开发者可以根据具体需求选择适合的方式。继承Thread类比较直接,但实现Runnable接口则可以实现更灵活的线程管理和资源共享。

引入匿名内部类和 Lambda 简化上述方法

匿名内部类

方法一

public class ThreadExample {public static void main(String[] args) {Thread thread = new Thread() {// 线程执行的代码@Overridepublic void run() {// 线程执行的代码for (int i = 0; i < 5; i++) {System.out.println("Runnable thread running: " + i);try {Thread.sleep(1000); // 暂停1秒} catch (InterruptedException e) {e.printStackTrace();}}}};thread.start();}
}

方法二

public class ThreadExample {public static void main(String[] args) {Thread thread = new Thread(new Runnable() {@Overridepublic void run() {// 线程执行的代码for (int i = 0; i < 5; i++) {System.out.println("Runnable thread running: " + i);try {Thread.sleep(1000); // 暂停1秒} catch (InterruptedException e) {e.printStackTrace();}}}});thread.start();}
}

   Lambda 表达式

public class ThreadExample {public static void main(String[] args) {Thread thread = new Thread(() -> {// 线程执行的代码for (int i = 0; i < 5; i++) {System.out.println("Runnable thread running: " + i);try {Thread.sleep(1000); // 暂停1秒} catch (InterruptedException e) {e.printStackTrace();}}});thread.start();}
}

Lambda 表达式由于简洁,所以是日常开发中常用的方法。

查看线程

当我们创建好一个线程后如何查看线程的状态呢?

使用 jconsole 命令观察线程

我们打开 jdk 文件夹所在目录,找到 bin 文件夹。

找到 jconsole.exe 双击打开。

 

 选择我们的类名的选项,点击连接。

 连接好了后,选择线程,然后找到 Thread-0 这个就是我们手动创建的线程,我们可以查看该线程的运行情况。

 Thread 中常见的方法

Thread类是Java中用于创建和管理线程的重要类,提供了多种方法来控制线程的行为和状态。以下是一些常见的Thread类方法:

  1. **start()**:启动线程,JVM会调用线程的run()方法。

  2. **run()**:线程执行的代码逻辑所在的方法。可以被重写来定义线程的任务。

  3. **sleep(long millis)**:使当前线程暂停指定的时间(毫秒),在此期间线程不会执行。

  4. **join()**:等待调用该方法的线程完成后再继续执行。这是实现线程间的同步的一种方式。

  5. **interrupt()**:中断线程,设置线程的中断状态。如果该线程正在阻塞(例如在sleep()wait()中),则会抛出InterruptedException

  6. **isAlive()**:判断线程是否仍在运行中,返回true表示线程处于活动状态。

  7. **getName()**:返回线程的名称。

  8. **setName(String name)**:设置线程的名称。

  9. **getPriority()**:返回线程的优先级。

  10. **setPriority(int priority)**:设置线程的优先级,优先级范围为Thread.MIN_PRIORITY(1)到Thread.MAX_PRIORITY(10)。

  11. **yield()**:提示调度器当前线程愿意让出对 CPU 的占用,由其他同等或更高优先级的线程获得执行机会。

  12. **currentThread()**:静态方法,返回对当前正在执行的线程对象的引用。

示例:

以下是一个简单的代码示例,演示了部分常见方法的用法:

class MyThread extends Thread {@Overridepublic void run() {System.out.println(getName() + " is running");try {sleep(1000); // 暂停1秒} catch (InterruptedException e) {System.out.println(getName() + " was interrupted");}System.out.println(getName() + " has finished running");}
}public class ThreadMethodsExample {public static void main(String[] args) {MyThread thread1 = new MyThread();MyThread thread2 = new MyThread();thread1.setName("Thread-1");thread2.setName("Thread-2");thread1.start();thread2.start();try {thread1.join(); // 等待thread1完成thread2.join(); // 等待thread2完成} catch (InterruptedException e) {e.printStackTrace();}System.out.println("All threads have finished executing.");}
}

在这个示例中,我们创建了两个线程,设置了它们的名称,并演示了start()sleep()join()方法的使用。

如何中断一个线程

在Java中,手动中断一个线程的原理主要依赖于线程的 interrupt() 方法和线程的 isInterrupted() 状态。通过调用一个线程的 interrupt() 方法,可以设置该线程的中断状态为 true。这通常用于通知线程它应该停止当前的工作,并进行清理或其他的收尾操作。

当一个线程被中断后,如果该线程在阻塞状态(例如,等待输入、休眠等),则会抛出 InterruptedException。如果线程在其他执行状态中,通常需要在合适的位置检查该线程的中断状态,决定是否需要停止执行。

以下是一个简单的案例,展示如何手动中断一个线程:

示例代码

class MyRunnable implements Runnable {@Overridepublic void run() {try {System.out.println("线程开始工作...");// 模拟长时间工作的情况for (int i = 0; i < 10; i++) {// 检查线程是否被中断if (Thread.currentThread().isInterrupted()) {System.out.println("线程被中断,退出工作...");return; // 退出运行}System.out.println("工作中: " + i);// 模拟工作过程中的延时Thread.sleep(1000);}} catch (InterruptedException e) {// 如果线程因为sleep被中断,会抛出InterruptedExceptionSystem.out.println("线程被中断,捕获到异常: " + e.getMessage());} finally {System.out.println("线程清理工作,准备结束...");}}
}public class ThreadInterruptExample {public static void main(String[] args) throws InterruptedException {Thread myThread = new Thread(new MyRunnable());myThread.start();// 主线程等待2秒,然后中断myThreadThread.sleep(2000);System.out.println("主线程请求中断myThread...");myThread.interrupt(); // 中断线程// 等待myThread结束myThread.join();System.out.println("主线程结束。");}
}

代码解释

  1. MyRunnable 类实现 Runnable 接口并重写 run() 方法。
  2. 在 run() 方法中,模拟了一个长时间工作的循环,每次循环检查线程的中断状态。
  3. 如果线程被中断,可以通过 Thread.currentThread().isInterrupted() 方法来检测,并通过返回来优雅地退出工作。
  4. 在主线程中,创建并启动一个新线程,等待2秒后调用 interrupt() 方法中断它。
  5. 如果在 sleep() 等待期间线程被中断,InterruptedException 将被抛出,因此可以在 catch 块中进行相应处理。
  6. 最后,使用 join() 等待 myThread 完成所有操作后再结束主线程。

总结

通过这个案例,可以清晰地看到如何手动中断线程以及如何处理线程的中断状态。这种机制在多线程编程中非常重要,确保了线程可以在合适的时机响应中断请求,从而实现更好的资源管理和程序健壮性。

join 等待一个线程

在Java中,等待一个线程的原理主要依赖于 Thread 类中的 join() 方法。调用 join() 方法可以使当前线程(即调用 join() 的线程)等待另一个线程完成执行。原理是通过线程的状态管理,使得调用 join() 的线程在被调用线程执行完之前不会继续执行,确保线程之间的执行顺序。

示例代码

下面是一个简单的案例,展示如何等待一个线程的完成:

class MyRunnable implements Runnable {@Overridepublic void run() {System.out.println("线程 " + Thread.currentThread().getName() + " 开始工作...");try {// 模拟长时间的工作Thread.sleep(2000); // 休眠2秒} catch (InterruptedException e) {System.out.println("线程被中断: " + e.getMessage());}System.out.println("线程 " + Thread.currentThread().getName() + " 工作完成。");}
}public class ThreadJoinExample {public static void main(String[] args) {Thread myThread = new Thread(new MyRunnable(), "MyThread");myThread.start(); // 启动线程try {System.out.println("主线程等待 " + myThread.getName() + " 完成...");myThread.join(); // 等待 myThread 完成} catch (InterruptedException e) {System.out.println("主线程被中断: " + e.getMessage());}System.out.println("主线程继续执行,已等待 " + myThread.getName() + " 完成。");}
}

代码解释

  1. MyRunnable 类:实现 Runnable 接口并重写 run() 方法。在 run() 方法中,我们模拟了一个长时间的工作,使用 Thread.sleep(2000) 使线程休眠 2 秒。
  2. 主类 ThreadJoinExample
    • 创建一个新的线程 myThread,并将 MyRunnable 实例作为参数传入。
    • 启动线程 myThread,这会调用其 run() 方法。
    • 在主线程中,调用 myThread.join(),这将使主线程等待 myThread 完成执行。
    • 一旦 myThread 执行完成,主线程将继续执行,并打印出相应的信息。

总结

通过这个案例,可以看到如何使用 join() 方法来等待线程的完成。这样可以有效地控制线程的执行顺序,确保在某些操作完成后再进行后续处理。例如,在多个线程之间需要协调工作时,使用 join() 使得某些操作依赖于另一个线程的完成,可以减少潜在的竞争和数据不一致问题。这种机制在多线程编程中非常重要,特别是在任务依赖的场景下。

获取线程的状态

在Java中,线程的状态主要有以下几种:

  1. NEW(新建):当线程被创建但尚未开始运行时,处于此状态。
  2. RUNNABLE(可运行):线程可以运行,也可能正在运行。这并不一定意味着线程正在执行,因为线程调度可能把该线程挂起。
  3. BLOCKED(阻塞):线程在等待一个监视器锁时被阻塞,无法继续执行。
  4. WAITING(等待):线程在等待另一个线程执行特定动作时进入此状态,如等待锁的释放。
  5. TIMED_WAITING(计时等待):线程在等待特定时间段内的某个条件时进入此状态。
  6. TERMINATED(终止):线程完成执行或因异常退出后进入此状态。

如何获取线程状态

可以使用 Thread 类的 getState() 方法来获取线程的当前状态。下面是一个示例代码,展示了如何创建线程并获取其状态:

public class ThreadStateExample {public static void main(String[] args) {Thread thread = new Thread(new MyRunnable());// 获取并打印状态:NEWSystem.out.println("Thread state after creation: " + thread.getState());thread.start();// 获取并打印状态:RUNNABLESystem.out.println("Thread state after starting: " + thread.getState());try {// 主线程等待一段时间以确保子线程有机会运行Thread.sleep(100);} catch (InterruptedException e) {e.printStackTrace();}// 获取并打印状态:TERMINATEDSystem.out.println("Thread state after completion: " + thread.getState());}static class MyRunnable implements Runnable {@Overridepublic void run() {// 子线程正在运行try {// 模拟一些工作Thread.sleep(200);} catch (InterruptedException e) {e.printStackTrace();}}}
}

说明

  1. 创建线程:在创建 Thread 对象后,可以通过 getState() 获取其状态,初始状态为 NEW
  2. 启动线程:调用 start() 方法后,线程状态可能变为 RUNNABLE
  3. 等待:主线程通过 sleep() 方法等待,使子线程有机会执行。
  4. 观察状态:通过 getState() 方法可以随时获取线程的状态。

这种方式能够有效地跟踪线程的状态变化,以便在多线程编程中进行调试和控制。

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

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

相关文章

1.1 计算机网络基本概述

欢迎大家订阅【计算机网络】学习专栏&#xff0c;开启你的计算机网络学习之旅&#xff01; 文章目录 前言一、网络的基本概念二、集线器、交换机和路由器三、互连网与互联网四、网络的类型五、互连网的组成1. 边缘部分2. 核心部分 六、网络协议 前言 计算机网络是现代信息社会…

LVGL学习

注&#xff1a;本文使用的lvgl-release-v8.3版本&#xff0c;其它版本可能稍有不同。 01 快速入门 1.1 LVGL模拟器配置 day01-02_课程介绍_哔哩哔哩_bilibili LVGL开发教程 (yuque.com) 如果按照上述视频和文档中配置不成功的话&#xff0c;直接重装VsCode&#xff0c;我的…

java实现系统文件管理

java实现系统文件管理 环境&#xff1a;jdk17springbootVueElementUI 背景&#xff1a;公司所做的项目需要别的系统向我们服务器上传文件&#xff0c;当我们需要查看这些文件什么时候上传的、文件数据是怎样的&#xff0c;只能去机房&#xff0c;排查问题效率较低&#xff0c;…

【VitualBox】VitualBox的网络模式+网络配置

VirtualBox 1. 简介 VirtualBox 是一款开源虚拟机软件&#xff0c;使用者可以在VirtualBox上安装并且执行Solaris、Windows、DOS、Linux、OS/2 Warp、BSD等系统作为客户端操作系统。 2. 六种网络接入模式 VirtualBox提供了多种网络接入模式&#xff0c;他们各有优缺点&#xf…

Baumer工业相机堡盟工业相机如何通过NEOAPI SDK获取相机当前数据吞吐量(Python)

Baumer工业相机堡盟工业相机如何通过NEOAPI SDK里函数来获取相机当前数据吞吐量&#xff08;Python&#xff09; Baumer工业相机Baumer工业相机的数据吞吐量的技术背景CameraExplorer如何查看相机吞吐量信息在NEOAPI SDK里通过函数获取相机接口吞吐量 Baumer工业相机通过NEOAPI…

移情别恋c++ ദ്ദി˶ー̀֊ー́ ) ——13.mapset

1. 关联式容器 在初阶阶段&#xff0c;我们已经接触过STL中的部分容器&#xff0c;比如&#xff1a;vector、list、deque、 forward_list(C11)等&#xff0c;这些容器统称为序列式容器&#xff0c;因为其底层为线性序列的数据结构&#xff0c;里面 存储的是元素本身。那什么是关…

【与C++的邂逅】--- 类和对象(上)

Welcome to 9ilks Code World (๑•́ ₃ •̀๑) 个人主页: 9ilk (๑•́ ₃ •̀๑) 文章专栏&#xff1a; 与C的邂逅 本篇博客将讲解C中的类和对象&#xff0c;C是面向对象的语言&#xff0c;面向对象三大特性是封装,继承,多态。学习类和对象&#xff0c;我们可…

【C语言】深入讲解指针(中)

文章目录 前言函数指针函数指针变量的创建函数指针变量的使用两段有趣的代码typedef 关键字 函数指针数组函数指针的使用最后 前言 上一章深入讲解指针&#xff08;上&#xff09;我们对字符指针、数组指针、指针和数组传参进行了讲解&#xff0c;本章将对函数指针进行讲解&am…

【Python】标准库的使用

文章目录 标准库日期计算字符串操作剑指offer 58&#xff0c;翻转单词顺序思路 leetcode 796&#xff0c;旋转字符串思路 leetcode 2255&#xff0c;统计是给定字符串前缀的字符串数目思路 文件查找工具 Python 通过模块来体现“库” 降低了程序猿的学习成本提高了程序的开发效…

【C语言篇】编译和链接以及预处理介绍(下篇)

文章目录 前言#和###运算符##运算符 命名约定#undef命令⾏定义条件编译#if和#endif多个分支的条件编译判断是否被定义嵌套指令 头文件被包含头文件被包含的方式本地文件包含库文件的包含 嵌套文件包含 其他预处理指令 写在最后 前言 本篇接前一篇【C语言篇】编译和链接以及预处…

【LeetCode】每日一题 2024_9_16 公交站间的距离(模拟)

前言 每天和你一起刷 LeetCode 每日一题~ LeetCode 启动&#xff01; 题目&#xff1a;公交站间的距离 代码与解题思路 func distanceBetweenBusStops(distance []int, start int, destination int) int {// 首先让 start > destination, 这两个谁大对结果没有影响&#…

免费爬虫软件“HyperlinkCollector超链采集器v0.1”

HyperlinkCollector超链采集器单机版v0.1 软件采用python的pyside2和selenium开发,暂时只支持window环境&#xff0c;抓取方式支持普通程序抓取和selenium模拟浏览器抓取。软件遵守robots协议。 首先下载后解压缩&#xff0c;然后运行app目录下的HyperlinkCollector.exe 运行…

C语言——rand函数

一、rand函数 这是一个在 C 标准库 <stdlib.h> 中定义的函数&#xff0c;用于生成伪随机数&#xff0c;默认情况下&#xff0c;它生成从 0 到 RAND_MAX 的伪随机数&#xff0c;其中 RAND_MAX 是一个常数&#xff0c;通常是 32767。 1、函数原型&#xff1a; 2、函数返回…

【数据结构】排序算法---直接插入排序

文章目录 1. 定义2. 算法步骤3. 动图演示4. 性质5. 算法分析6. 代码实现C语言PythonJavaCGo 7. 折半插入排序代码实现——C 结语 1. 定义 直接插入排序是一种简单直观的排序算法。它的工作原理为将待排列元素划分为「已排序」和「未排序」两部分&#xff0c;每次从「未排序的」…

C++ char*和char[] 可能指向的内存区域详解(附实验)

C char* 指向的内存区域详解 写在前面c内存结构简介指针常量和常量指针简介情况一&#xff1a;char* 指向栈区内容情况二&#xff1a;char* 指向堆区内容情况三&#xff1a;char* 指向常量区内容情况四&#xff1a;char* 指向静态区内容情况五&#xff1a;char* 指向全局区内容…

SQL题目分析:打折日期交叉问题--计算品牌总优惠天数

在电商平台的数据分析中&#xff0c;处理品牌促销活动的日期交叉问题是一个挑战。本文将介绍几种高级SQL技巧&#xff0c;用于准确计算每个品牌的总优惠天数&#xff0c;即使在存在日期交叉的情况下。 问题背景 我们有一个促销活动表 shop_discount&#xff0c;记录了不同品牌…

docker-compose 部署 flink

下载 flink 镜像 [rootlocalhost ~]# docker pull flink Using default tag: latest latest: Pulling from library/flink 762bedf4b1b7: Pull complete 95f9bd9906fa: Pull complete a880dee0d8e9: Pull complete 8c5deab9cbd6: Pull complete 56c142282fae: Pull comple…

【二叉树进阶】二叉搜索树

目录 1. 二叉搜索树概念 2. 二叉搜索树的实现 2.1 创建二叉搜索树节点 2.2 创建实现二叉搜索树 2.3 二叉搜索树的查找 2.4 二叉搜索树的插入 2.5 二叉搜索树的删除 2.6 中序遍历 2.7 完整代码加测试 3. 二叉搜索树的应用 3.1 K模型&#xff1a; 3.2 KV模型&#xf…

【C++11】可变参数模板

【C11】可变参数模板 一、可变参数模板概念以及定义方式 ​ 在C11之前&#xff0c;类模板和函数模板只能含有固定数量的模板参数&#xff0c;c11增加了可变模板参数特性&#xff1a;允许模板定义中包含0到任意个模板参数。声明可变参数模板时&#xff0c;需要在typename或cla…

数据在内存中的存储方式

前言&#xff1a;已经好久没更新了&#xff0c;开学之后学习编程的时间少了很多。因此&#xff0c;已经好几个礼拜都没有写文章了。 在讲解操作符的时候&#xff0c;我们就已经学习过了整数在内存中的存储方式。若有不懂的伙伴可以前往操作符详解进行学习。今天我们主要来学习…