探索多线程编程:线程的本质、状态和属性

news2025/1/11 2:54:56

目录

  • 什么是线程
  • 线程状态
    • 新建线程
    • 可运行线程
    • 阻塞和等待线程
    • 终止线程
  • 线程属性
    • 优先级
    • 线程名
    • 守护线程
    • 中断线程
    • 未捕获异常的处理器

在现代计算机编程中,多线程是一个重要而强大的概念。它使得我们能够更有效地利用多核处理器、提高程序性能并实现并发操作。

什么是线程

线程是程序执行的最小单元,它是操作系统调度的基本单位。与进程不同,线程共享相同的进程内存空间,这意味着它们可以访问相同的数据和资源。线程之间的通信更加轻松,但也需要更仔细的同步,以避免竞态条件和数据冲突。
每个任务在一个线程中执行,线程是控制线程的简称。一个程序同时运行多个线程,就可以称这个程序是多线程的。
我们来写一个多线程程序看一下执行结果:

public class main {
    public static void main(String[] args) {
        //构造一个新线程
        Thread t1 = new Thread(new Runnable() {
            @Override
            public void run() {
                for (int i = 0; i < 200; i++) {
                    System.out.println("这是第一个线程!");
                }
            }
        });

        Thread t2 = new Thread(new Runnable() {
            @Override
            public void run() {
                for (int i = 0; i < 200; i++) {
                    System.out.println("这是第二个线程!");
                }
            }
        });
        System.out.println("线程1启动");
        //启动新线程
        t1.start();
        System.out.println("线程2启动");
        t2.start();
    }
}

在这里插入图片描述
通过结果我们可以看到两个线程是同时执行的。
创建线程的几种方法可以看完之前写的博客。

线程状态

线程有如下6种状态:

  • New(新建)
  • Runnable(可运行)
  • Blocked(阻塞)
  • Waiting(等待)
  • Time waiting(计时等待)
  • Terminated(终止)

新建线程

用new操作符创建一个新线程时,这个线程还没有开始运行。
例如:

        //构造一个新线程
        Thread t1 = new Thread(new Runnable() {
            @Override
            public void run() {
                for (int i = 0; i < 200; i++) {
                    System.out.println("这是第一个线程!");
                }
            }
        });

可运行线程

当新创建的线程调用start方法后,就处于可运行状态。一个可运行的线程可能处于正在运行状态也可能没有运行。这是因为抢占式调度系统,给每一个可运行线程一个时间片来执行任务。每一个处理器运行一个线程,当线程的数目多于处理器数目时,调度器需要分配时间片,让多个线程并行运行。

阻塞和等待线程

当线程处于阻塞或者等待状态时,它是暂时不活动的。

  • 线程试图获取内部的对象锁,而这个锁被其他线程占有,以至线程处于阻塞状态。
  • 线程等待其他线程发出的通知或中断信号。例如调用Object.wait方法,Thread.join方法使线程进入等待状态。
  • 还有就是使用Thread.sleep,Object.wait,Thread.join方法使使线程进入计时等待状态。

例如:Thread.sleep(1000);使线程等待一秒。

终止线程

  • run方法正常退出,线程自然结束
  • 一个没有捕获的异常终止了run方法,使线程意外终止

我们可以通过getState()得到当前线程的状态。

线程属性

线程的属性是指线程的一些特征和配置选项,它们可以影响线程的行为和性能。

优先级

优先级用于确定线程在竞争CPU时间时的优先级,即决定了线程被调度执行的概率。每个线程都有一个默认优先级,通常为普通(Normal)优先级,范围从1(最低优先级)到10(最高优先级)。线程的优先级可以通过编程方式进行设置和调整。

线程的优先级决定了它在与其他线程竞争CPU时间时的优先顺序。操作系统的调度算法通常会考虑线程的优先级来决定将CPU时间分配给哪个线程。高优先级线程相对于低优先级线程来说,有更高的概率获得CPU时间片,即有更多的机会去执行任务。然而,优先级并不是绝对的,只是一个相对的概念,高优先级线程并非一定比低优先级线程先执行。

下面通过一个简单的例子来解释优先级:

public class test1 implements Runnable {
    public void run() {
        for (int i = 0; i < 5; i++) {
            System.out.println(Thread.currentThread().getName() + " is running");
        }
    }

    public static void main(String[] args) {
        Thread thread1 = new Thread(new test1());
        Thread thread2 = new Thread(new test1());

        thread1.setPriority(Thread.MIN_PRIORITY); // 设置线程1的优先级为最低
        thread2.setPriority(Thread.MAX_PRIORITY); // 设置线程2的优先级为最高

        thread1.start();
        thread2.start();
    }
}

在这里插入图片描述
在上述例子中,我们创建了两个线程 thread1 和 thread2,并将它们分别设置为最低优先级和最高优先级。在 run 方法中,每个线程会循环打印自己的名称。由于线程2的优先级更高,它具有更高的CPU时间片分配概率,因此它更有可能在竞争中先被调度执行。但是,该结果并不是绝对的,线程1仍然有机会在某些情况下先于线程2执行。

线程名

线程名用于为线程赋予一个可识别和标识的名称。每个线程都可以被命名,这样在调试和日志记录时就能更容易地识别和追踪线程的执行情况。线程名可以帮助我们区分不同的线程并理解它们在程序中的作用。
在Java中,可以通过setName方法为线程设置名字,也可以通过getName方法获取线程的名称。线程名通常是一个描述性的字符串,可以根据任务的特性和上下文进行命名,以便更好地反映线程的功能。

守护线程

守护线程(Daemon Thread)是指在程序运行过程中在后台默默执行的线程,它并不会阻止程序的退出。当所有非守护线程结束时,守护线程会自动终止。它们通常用于执行一些后台任务或提供一种服务性功能,而不需要干扰或阻塞主程序的正常执行。
守护线程的生命周期与程序的生命周期相伴而终。当所有非守护线程都结束时,守护线程会被自动停止,即使它们正在执行某些任务。为了标识一个线程为守护线程,可以使用线程对象的setDaemon(true)方法设置,这一方法必须在线程启动前调用

中断线程

中断线程是指在多线程编程中,一个线程被要求停止执行或被强制终止。这通常是通过发送一个中断信号或设置一个标志来实现的,让线程在接收到中断请求后,安全地停止执行并释放资源。中断线程的目的通常是为了避免线程无法正常退出或处理某些异常情况。
下面是一个简单的Java示例,演示了如何中断一个线程:


public class InterruptExample {

    public static void main(String[] args) {
        Thread myThread = new Thread(new MyRunnable());
        myThread.start(); // 启动线程

        // 主线程等待一段时间后中断myThread
        try {
            Thread.sleep(2000); // 等待2秒钟
            myThread.interrupt(); // 发送中断信号
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }

    static class MyRunnable implements Runnable {
        @Override
        public void run() {
            while (!Thread.currentThread().isInterrupted()) {
                System.out.println("线程正在执行...");
                try {
                    Thread.sleep(1000); // 线程每隔1秒钟执行一次
                } catch (InterruptedException e) {
                    // 捕获到中断信号后,线程会退出循环
                    System.out.println("线程被中断,即将退出...");
                    Thread.currentThread().interrupt(); // 重新设置中断状态
                }
            }
        }
    }
}

在这里插入图片描述

在上述示例中,我们创建了一个线程(myThread),然后在主线程中等待2秒钟后发送中断信号给myThread。在MyRunnable线程中,线程不断地执行,但会检查是否接收到中断信号。如果接收到中断信号,线程会在捕获到InterruptedException后退出循环,完成线程的中断操作。
需要注意的是,中断线程并不会强制终止线程,而是通过设置一个中断标志,让线程在合适的时候自行终止。这样可以确保线程在终止时能够完成必要的清理工作,以避免资源泄漏和不一致的状态

interrupted和isInterrupted,interrupted方法是一个静态方法,它检查当前线程是否被中断,并且会清除该线程的中断状态,isInterrupted是一个实例方法,用来检查是否有线程被中断。

未捕获异常的处理器

未捕获异常处理器(Uncaught Exception Handler)是在多线程或多线程应用程序中用来处理未捕获异常的一种机制。当一个线程抛出一个未捕获的异常时,如果该线程没有显式地捕获这个异常,那么异常会传递到线程的未捕获异常处理器中,这个处理器可以自定义,用于记录异常信息、执行特定的操作,或者进行应用程序的安全退出。
下面是一个简单的Java示例,演示了如何设置未捕获异常处理器:


public class UncaughtExceptionHandlerExample {

    public static void main(String[] args) {
        // 设置未捕获异常处理器
        Thread.setDefaultUncaughtExceptionHandler(new CustomExceptionHandler());

        // 创建一个线程并启动它,该线程会抛出一个未捕获异常
        Thread thread = new Thread(() -> {
            throw new RuntimeException("这是一个未捕获的异常");
        });
        thread.start();
    }

    static class CustomExceptionHandler implements Thread.UncaughtExceptionHandler {
        @Override
        public void uncaughtException(Thread t, Throwable e) {
            System.err.println("线程 " + t.getName() + " 抛出了未捕获的异常: " + e.getMessage());
            // 在这里可以添加自定义处理逻辑,如记录日志、发送通知等
        }
    }
}

在这里插入图片描述

在上述示例中,我们首先通过Thread.setDefaultUncaughtExceptionHandler方法设置了一个自定义的未捕获异常处理器(CustomExceptionHandler)。然后,创建了一个线程并启动它,该线程会故意抛出一个未捕获的异常。当这个异常抛出时,它会传递到我们设置的未捕获异常处理器中,CustomExceptionHandler中的uncaughtException方法将会执行,打印异常信息。您可以在这个方法中添加任何您需要的自定义处理逻辑,比如记录日志或发送通知。

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

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

相关文章

docker从零部署jenkins保姆级教程(下)

上一篇文章&#xff0c;我们完成了以下工作。 1)、docker部署jenkins 2)、建立第一个jenkins job 3)、通过jenkins job自动编译构建我们的github项目 上面所做的3个工作&#xff0c;其实都是为了这一篇文章打基础&#xff0c;不管是部署docker还是部署jenkins&#xff0c;我们最…

d435i 相机和imu标定

一、IMU 标定 使用 imu_utils 功能包标定 IMU&#xff0c;由于imu_utils功能包的编译依赖于code_utils&#xff0c;需要先编译code_utils&#xff0c;主要参考 相机与IMU联合标定_熊猫飞天的博客-CSDN博客 Ubuntu20.04编译并运行imu_utils&#xff0c;并且标定IMU_学无止境的…

补码:将减法运算转化为另一种形式的加法运算

文章目录 解析 个人见解&#xff0c;如有错误&#xff0c;请多包涵。 解析 对于人来说&#xff0c;减法是简单容易的。 被减数和减数列式相减&#xff0c;从低位到高位分别计算&#xff0c;有需要的借位就可以了。 这是一种可以在计算机上成立的理论方案&#xff0c;但是由于…

SpringMVC常用注解介绍及参数传递说明

前言 上一篇文章介绍了SpringMVC是什么以及它的工作流程和核心组件&#xff0c;介绍入门示例时&#xff0c;提到了RequestMapping注解&#xff0c;那么这篇文章就来介绍SpringMVC中更多的常用的注解&#xff0c;以及它的参数传递。 一. SpringMVC常用注解 1.1 RequestParam …

Homebrew安装cocoapods: zsh: command not found: brew解决方法

问题描述: 通过Homebrew安装cocoapods时,输入命令行 brew install cocoapods出现如下报错: zsh: command not found: brew zsh:找不到命令&#xff1a;brew 问题解决: 使用以下命令,重新安装Homebrew. /bin/zsh -c "$(curl -fsSL https://gitee.com/cunkai/HomebrewCN/ra…

永恒之黑_CVE-2020-0796漏洞复现

永恒之黑&#xff1a;CVE-2020-0796漏洞复现 目录 永恒之黑&#xff1a;CVE-2020-0796漏洞复现漏洞介绍漏洞影响范围漏洞复现1.环境准备2.复现过程 漏洞介绍 本漏洞源于SMBv3没有正确处理压缩的数据包&#xff0c;在解压数据包的时候使用客户端传过来的长度进行解压时&#xf…

机器学习——boosting之提升树(未完)

提升树和adaboost基本流程是相似的 我看到提升树的时候&#xff0c;懵了 这…跟adaboost有啥区别&#xff1f;&#xff1f;&#xff1f; 直到看到有个up主说了&#xff0c;我才稍微懂 相当于&#xff0c;我在adaboost里的弱分类器&#xff0c;换成CART决策树就好了呗&#xff1…

springboot + activiti实现activiti微服务化

概述 本文介绍如何将springbootactiviti进行整合,并配合eureka,zuul和feign实现activiti的微服务化,将流程控制和业务逻辑分离. 并实现了几个比较特殊的功能,比如时间段委托(某人请假或出差,出差时间内,所有待办交给被委托人处理),比如节点的无限级加签功能(流程本身有不确定性…

php 获取今日、昨日、上周、本月的起始时间戳和结束时间戳的方法非常简单

php 获取今日、昨日、上周、本月的起始时间戳和结束时间戳的方法&#xff0c;主要使用到了 php 的时间函数 mktime。下面首先还是以示例说明如何使用 mktime 获取今日、昨日、上周、本月的起始时间戳和结束时间戳&#xff0c;然后在介绍一下 mktime 函数作用和用法。非常简单哦…

vue3组件通信学习笔记

1、Prop 父组件 <template><div class"parent"><h1>我是父元素</h1><Child :msg"msg"></Child></div> </template><script setup> import Child from ./Child.vue let msg ref(我是父组件的数据…

单片机采集传感器数据(整形,浮点型)modbus上传

浮点型数据 占两个寄存器&#xff08;四个字节&#xff09; short 整形 占一个寄存器 &#xff08;两个字节&#xff09; 注意&#xff01;&#xff01;&#xff01;&#xff01; stm32 是小端模式&#xff0c;而modbus解析数据是大端模式 所以先发送高字节 如int a16777220…

【QML】使用 QtQuick2的ListView创建一个列表(一)

qtquick2版本和qtquick1版本分别提供了一个ListView组件供使用&#xff0c;两个组件在使用上和属性的提供上还是有比较大的差异的&#xff0c;因为qtquick2是新的&#xff0c;所以就以改版本为基础学习如何使用&#xff1b; 首先&#xff0c;要知道ListView提供了那些属性提供修…

2023年智能家居占消费电子出货量28%,蓝牙Mesh照明占据重要位置

市场研究机构 TechInsights 的最新报告显示&#xff0c;预计 2023 年全球消费者在智能家居相关硬件、服务和安装费方面的支出将复苏&#xff0c;达到 1310 亿美元&#xff0c;比 2022 年增长 10%。TechInsights 表示&#xff0c;消费者在智能家居系统和服务上的支出将继续强劲增…

【UIPickerView案例05-省市选择界面数据展示 Objective-C语言】

一、省市选择界面数据展示 1.省市选择界面数据展示,就是这样的一个东西 我们接下来,看我们第二个案例,就是这个省市选择, 左边选择一个省,右边就把这个省所有的市展示出来 比如,我现在展示的是山东的城市, 我选择一个山西 第一步干嘛,是不是也是分析它的界面 1)上…

Android Automotive编译

系统准备 安装系统 准备一台安装Ubuntu系统的机器&#xff08;windows系统的机器可以通过WSL安装ubuntu系统&#xff09; 安装docker 本文使用docker进行编译&#xff0c;因此提前安装docker。参考网络链接安装docker并设置为不使用sudo进行docker操作。 参考链接&#xff…

B-Tree 索引和 Hash 索引的对比

分析&回答 B-Tree 索引的特点 B-tree 索引可以用于使用 , >, >, <, < 或者 BETWEEN 运算符的列比较。如果 LIKE 的参数是一个没有以通配符起始的常量字符串的话也可以使用这种索引。 有时&#xff0c;即使有索引可以使用&#xff0c;MySQL 也不使用任何索引。…

2023 最新 Git 分布式版本控制系统介绍和下载安装使用教程

Git 基本概述 Git 是一个开源的分布式版本控制系统&#xff0c;用于敏捷高效地处理任何或大或小的项目。 集中式和分布式的区别&#xff1f; 最常见的集中式版本控制系统是SVN&#xff0c;版本库是集中放在中央处理器中的&#xff0c;而干活的时候&#xff0c;用的都是自己电…

类和对象(Java)

目录 一、面向对象的初步认知1、什么是面向对象2、面向对象与面向过程 二、类和类的实例化1、什么是类2、类的实例化3、类和对象的说明 三、this引用1、为什么要有this引用2、什么是this引用3、this引用的特性 四、对象的构造及初始化1、如何初始化对象2、构造方法 五、封装1、…

React三属性之:props

作用 将父组件的参数传递给子组件 父组件 import ./App.css; import React from react; import PropsTest from ./pages/propsTest class App extends React.Component{render(){return(<div><h2>App组件</h2><PropsTest obj{{name:王惊涛,age:27}}>…

首发丨全球首款用于激光雷达的商用光控超表面芯片发布!激光雷达降本再添可选项

《激光雷达老炮儿》最新获悉,美国光学半导体创业公司Lumotive于上周五宣布正式对外推出其首款极具开创性、基于光控超表面 (LCM)技术的完整产品LM10,该产品也是世界上首款商用数字光束控制解决方案。 与机械系统相比,Lumotive的数字光束控制技术凭借其卓越的成本、尺寸和可…