Java线程的创建和使用

news2024/11/23 18:25:40

目录

基本概念

线程的创建和使用

线程的生命周期


基本概念

程序(program)

        程序(program)是为完成特定任务、用某种语言编写的一组指令的集合。即指一段静态的代码,静态对象。

进程(process)

        进程(process)是程序的一次执行过程,或是正在运行的一个程序。是一个动态的过程(生命周期):有它自身的产生、存在和消亡的过程。

        程序是静态的,进程是动态的。进程作为资源分配的单位,系统在运行时会为每个进程分配不同的内存区域。

线程(thread)

        进程可进一步细化为线程,是一个程序内部的一条执行路径。若一个进程同一时间并行执行多个线程,就是支持多线程的。

        线程作为调度和执行的单位,每个线程拥有独立的运行栈和程序计数器(pc),线程切换的开销小。

        一个进程中的多个线程共享相同的内存单元/内存地址空间它们从同一堆中分配对象,可以访问相同的变量和对象。这就使得线程间通信更简便、高效。但多个线程操作共享的系统资源可能就会带来安全的隐患。

多线程

        Java 给多线程编程提供了内置的支持。 一条线程指的是进程中一个单一顺序的控制流,一个进程中可以并发多个线程,每条线程并行执行不同的任务。

单核CPU和多核CPU

        单核CPU,其实是一种假的多线程,因为在一个时间单元内,也只能执行一个线程的任务。例如:虽然有多车道,但是收费站只有一个工作人员在收费,只有收了费才能通过,那么CPU就好比收费人员。如果有某个人不想交钱,那么收费人员可以把他“挂起”(晾着他,等他想通了,准备好了钱,再去收费)。但是因为CPU时间单元特别短,因此感觉不出来。

        如果是多核的话,才能更好的发挥多线程的效率(现在的服务器都是多核的)。

       一个Java应用程序java.exe,其实至少有三个线程:main()主线程,gc()垃圾回收线程,异常处理线程。当然如果发生异常,会影响主线程。 

并行与并发

        并行:多个CPU同时执行多个任务。比如:多个人同时做不同的事。 
        并发:一个CPU(采用时间片)同时执行多个任务。比如:秒杀、多个人做同一件事。

多线程程序的优点

        1. 提高应用程序的响应。对图形化界面更有意义,可增强用户体验。
        2. 提高计算机系统CPU的利用率。
        3. 改善程序结构。将既长又复杂的进程分为多个线程,独立运行,利于理解和修改。

何时需要多线程

        程序需要同时执行两个或多个任务。
        程序需要实现一些需要等待的任务时,如用户输入、文件读写操作、网络操作、搜索等。
        需要一些后台运行的程序时。

线程的分类

        Java中的线程分为两类:一种是守护线程,一种是用户线程。它们在几乎每个方面都是相同的,唯一的区别是判断JVM何时离开。

        守护线程是用来服务用户线程的,通过在start()方法前调用thread.setDaemon(true)可以把一个用户线程变成一个守护线程。

        Java垃圾回收就是一个典型的守护线程。若JVM中都是守护线程,当前JVM将退出。

线程的创建和使用

线程的创建和启动

Java语言的JVM允许程序运行多个线程,它通过java.lang.Thread类来体现。

每个线程都是通过某个特定Thread对象的run()方法来完成操作的,经常把run()方法的主体称为线程体。

通过该Thread对象的start()方法来启动这个线程,而非直接调用run()。

Thread类

构造器

序号构造方法及描述
1Thread()
创建新的Thread对象。
2Thread(String threadname)
创建线程并指定线程实例名。
3Thread(Runnable target)
指定创建线程的目标对象,它实现了Runnable接口中的run方法。
4Thread(Runnable target, String name)
创建新的Thread对象。

方法(被 Thread 对象调用的)

序号方法描述
1public void start()
使该线程开始执行;Java 虚拟机调用该线程的 run 方法。
2public void run()
如果该线程是使用独立的 Runnable 运行对象构造的,则调用该 Runnable 对象的 run 方法;否则,该方法不执行任何操作并返回。
3public final void setName(String name)
改变线程名称,使之与参数 name 相同。
4public final void setPriority(int priority)
 更改线程的优先级。
5public final void setDaemon(boolean on)
将该线程标记为守护线程或用户线程。
6public final void join(long millisec)
等待该线程终止的时间最长为 millis 毫秒。
7public void interrupt()
中断线程。
8public final boolean isAlive()
测试线程是否处于活动状态。

静态方法(被Thread 类调用的)

序号方法描述
1public static void yield()
暂停当前正在执行的线程对象,并执行其他线程。
2public static void sleep(long millisec)
在指定的毫秒数内让当前正在执行的线程休眠(暂停执行),此操作受到系统计时器和调度程序精度和准确性的影响。
3public static boolean holdsLock(Object x)
当且仅当当前线程在指定的对象上保持监视器锁时,才返回 true。
4public static Thread currentThread()
返回对当前正在执行的线程对象的引用。
5public static void dumpStack()
将当前线程的堆栈跟踪打印至标准错误流。

​​​​​​​创建线程的四种方式

JDK1.5之前创建新执行线程有两种方法:继承Thread类的方式 和 实现Runnable接口的方式。

JDK5.0 新增线程创建方式:实现Callable接口的方式 和 使用线程池。

方式一:继承Thread类

1) 定义子类继承Thread类。
2) 子类中重写Thread类中的run方法。
3) 创建Thread子类对象,即创建了线程对象。
4) 调用线程对象start方法:启动线程,调用run方法。

class MyThread extends Thread {
    public MyThread(){
        super();
    }
    public void run(){
        for(int i = 0; i < 100; i++) {
            System.out.println("子线程: " + i)
        }
    }
}
public class TestThread {
    public static void main(String[] args) {
        //1.创建线程。
        MyThread mt, = Tew MyThread();
        //2.启动线程并调用当前线程的run()方法。
        mt. start();
    }
}

 注意点:

        1. 如果自己手动调用run()方法,那么就只是普通方法,没有启动多线程模式。
        2. run()方法由JVM调用,什么时候调用,执行的过程控制都有操作系统的CPU调度决定。
        3. 想要启动多线程,必须调用start方法。
        4. 一个线程对象只能调用一次start()方法启动,如果重复调用了,则将抛出以上的异常“IllegalThreadStateException”。

方式二:实现Runnable接口

1) 定义子类,实现Runnable接口。
2) 子类中重写Runnable接口中的run方法。
3) 通过Thread类含参构造器创建线程对象。
4) 将Runnable接口的子类对象作为实际参数传递给Thread类的构造器中。
5) 调用Thread类的start方法:开启线程,调用Runnable子类接口的run方法。

public class DisplayMessage implements Runnable {
    private String message;
   
    public DisplayMessage(String message) {
        this.message = message;
    }
   
    public void run() {
        while(true) {
            System.out.println(message);
        }
    }
}

继承方式和实现方式的联系与区别

public class Thread extends Object implements Runnable

区别:
        继承Thread:线程代码存放Thread子类run方法中。
        实现Runnable:线程代码存在接口的子类的run方法。

实现方式的好处:
        避免了单继承的局限性。
        多个线程可以共享同一个接口实现类的对象,非常适合多个相同线程来处理同一份资源。

方式三:实现Callable接口

1. 创建 Callable 接口的实现类,并实现 call() 方法,该 call() 方法将作为线程执行体,并且有返回值。
2. 创建 Callable 实现类的实例,使用 FutureTask 类来包装 Callable 对象,该 FutureTask 对象封装了该 Callable 对象的 call() 方法的返回值。
3. 使用 FutureTask 对象作为 Thread 对象的 target 创建并启动新线程。
4. 调用 FutureTask 对象的 get() 方法来获得子线程执行结束后的返回值。

与使用Runnable相比, Callable功能更强大些;相比run()方法,可以有返回值;方法可以抛出异常;支持泛型的返回值;需要借助FutureTask类,比如获取返回结果;

Future接口

        可以对具体Runnable、Callable任务的执行结果进行取消、查询是否完成、获取结果等。
        FutrueTask是Futrue接口的唯一的实现类。
        FutureTask 同时实现了Runnable, Future接口。它既可以作为Runnable被线程执行,又可以作为Future得到Callable的返回值。

public class CallableThreadTest implements Callable<Integer> {
    public static void main(String[] args) {  
        CallableThreadTest ctt = new CallableThreadTest();  
        FutureTask<Integer> ft = new FutureTask<>(ctt);  
        for(int i = 0;i < 100;i++) {  
            System.out.println(Thread.currentThread().getName()+" 的循环变量i的值"+i);  
            if(i==20)  
            {  
                new Thread(ft,"有返回值的线程").start();  
            }  
        }  
        try {  
            System.out.println("子线程的返回值:"+ft.get());  
        } catch (InterruptedException e) {  
            e.printStackTrace();  
        } catch (ExecutionException e) {  
            e.printStackTrace();  
        }  
  
    }
    @Override  
    public Integer call() throws Exception {  
        int i = 0;  
        for(;i<100;i++) {  
            System.out.println(Thread.currentThread().getName()+" "+i);  
        }  
        return i;  
    }  
}

方式四:使用线程池

背景:
        经常创建和销毁、使用量特别大的资源,比如并发情况下的线程,对性能影响很大。  

思路:
        提前创建好多个线程,放入线程池中,使用时直接获取,使用完放回池中。可以避免频繁创建销毁、实现重复利用。类似生活中的公共交通工具。

好处:
        提高响应速度(减少了创建新线程的时间);
        降低资源消耗(重复利用线程池中线程,不需要每次都创建);
        便于线程管理;
        corePoolSize:核心池的大小;
        maximumPoolSize:最大线程数  keepAliveTime:线程没有任务时最多保持多长时间后会终止;

线程池相关API

JDK 5.0起提供了线程池相关API:ExecutorService 和 Executors。

ExecutorService:真正的线程池接口。常见子类ThreadPoolExecutor。

序号方法及描述
1void execute(Runnable command)
执行任务/命令,没有返回值,一般用来执行Runnable。
2<T> Future<T> submit(Callable<T> task)
执行任务,有返回值,一般又来执行Callable。
3void shutdown()
关闭连接池。

Executors:工具类、线程池的工厂类,用于创建并返回不同类型的线程池。

序号方法及描述
1Executors.newCachedThreadPool()
创建一个可根据需要创建新线程的线程池。
2Executors.newFixedThreadPool(n)
创建一个可重用固定线程数的线程池。
3Executors.newSingleThreadExecutor()
创建一个只有一个线程的线程池。
4

Executors.newScheduledThreadPool(n)

创建一个线程池,它可安排在给定延迟后运行命令或者定期地执行。

线程的调度

调度策略:

        时间片:

        抢占式: 高优先级的线程抢占CPU

Java的调度方法:

        同优先级线程组成先进先出队列(先到先服务),使用时间片策略。
        对高优先级,使用优先调度的抢占式策略。

线程的优先级

每一个 Java 线程都有一个优先级,这样有助于操作系统确定线程的调度顺序。

Java 线程的优先级是一个整数,其取值范围是 1- 10 。默认情况下,每一个线程都会分配一个优先级 NORM_PRIORITY(5)。线程的优先级等级:

        MAX_PRIORITY:10 
        MIN _PRIORITY:1 
        NORM_PRIORITY:5 

涉及的方法:

        getPriority() :返回线程优先值
        setPriority(int newPriority) :改变线程的优先级

说明:

        线程创建时继承父线程的优先级;

        具有较高优先级的线程对程序更重要,并且应该在低优先级的线程之前分配处理器资源。但是, 低优先级只是获得调度的概率低,并非一定是在高优先级线程之后才被调用。线程优先级不能保证线程执行的顺序,而且非常依赖于平台。

线程的生命周期

JDK中用Thread.State类定义了线程的几种状态

要想实现多线程,必须在主线程中创建新的线程对象。Java语言使用Thread类及其子类的对象来表示线程,在它的一个完整的生命周期中通常要经历如下的五种状态:

        新建: 当一个Thread类或其子类的对象被声明并创建时,新生的线程对象处于新建状态。
        就绪:处于新建状态的线程被start()后,将进入线程队列等待CPU时间片,此时它已具备了运行的条件,只是没分配到CPU资源。
        运行:当就绪的线程被调度并获得CPU资源时,便进入运行状态, run()方法定义了线程的操作和功能。
        阻塞:在某种特殊情况下,被人为挂起或执行输入输出操作时,让出 CPU 并临时中止自己的执行,进入阻塞状态。
        死亡:线程完成了它的全部工作或线程被提前强制性地中止或出现异常导致结束。

线程的生命周期

 

线程状态转换图

 


 

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

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

相关文章

ESP32+MQTT+MySQL实现发布订阅【气味数据收集】

ESP32MQTTMySQL实现发布订阅【气味数据收集】 &#x1f52e;&#x1f52e;&#x1f52e;&#x1f52e;&#x1f52e;相关文章&#x1f52e;&#x1f52e;&#x1f52e;&#x1f52e;&#x1f52e; ESP32连接MQ Sensor实现气味反应 &#x1f517; https://blog.csdn.net/ws1516…

指针知多少

作者简介&#xff1a;დ旧言~&#xff0c;目前大一&#xff0c;现在学习Java&#xff0c;c&#xff0c;Python等 座右铭&#xff1a;松树千年终是朽&#xff0c;槿花一日自为荣。 望小伙伴们点赞&#x1f44d;收藏✨加关注哟&#x1f495;&#x1f495; ⛵前言 不知道大家还记…

avi格式怎么转换成mp4?教你几种简单转换小技巧

由于 AVI 格式的文件结构&#xff0c;它们不适合用于流式传输。这意味着&#xff0c;如果你想在线观看一个 AVI 格式的视频&#xff0c;你需要等待整个文件下载完成后才能开始观看。这对于那些希望尽快开始观看视频的人来说可能是一个问题。那么我们怎么将AVI格式的视频转换成M…

【C语言】memcpy,memmove,memcmp,memset函数详解

memcpy,memmove,memcmp,memset函数详解 memcpy函数一、 memcpy函数的定义&#xff1a;二、memcpy函数的功能&#xff1a;三、memcpy函数模拟memcpy注意事项 memmove函数一、memmove函数简介二、memmove函数的模拟1.两种情况2模拟实现 memcmp函数memecmp函数介绍 memset函数mems…

Cell 子刊 - 4D打印一只可变形的蜘蛛

国自然“十四五”优先发展领域公布&#xff0c;共计115项&#xff01;&#xff08;生物医学领域节选&#xff0c;近 50 项&#xff09;中提到 4D打印是一个重要方向。 凝胶状墨水使得3D打印应用于电子设备的金属物体更容易。 3D打印在经济性、设计自由度和效率方面均超越了传统…

Matplotlib中文乱码解决方案(两种方式)

Matplotlib 默认不支持中文字体&#xff0c;这因为 Matplotlib 只支持 ASCII 字符&#xff0c;但中文标注更加符合中国人的阅读习惯。因此&#xff0c;本节重点讲解如何在 Windows 环境下让 Matplotlib 显示中文。 Matplotlib中文乱码 当不对 Matplotlib 进行设置&#xff0c…

网页版五子棋设计实现自动化测试

目录 一、设计测试用例 二、执行测试 登录页面 功能测试 界面测试 注册界面 功能测试 界面测试 游戏大厅 功能测试 界面测试 游戏房间 功能测试 界面测试 一、设计测试用例 二、执行测试 在执行测试之前首先获取到驱动。 登录页面 功能测试 首先定义star…

2023年9月长沙/郑州/济南DAMA-CDGA/CDGP认证考试报名

据DAMA中国官方网站消息&#xff0c;2023年度第三期DAMA中国CDGA和CDGP认证考试定于2023年9月23日举行。 报名通道现已开启&#xff0c;相关事宜通知如下&#xff1a; 考试科目: 数据治理工程师(CertifiedDataGovernanceAssociate,CDGA) 数据治理专家(CertifiedDataGovernanc…

Spark(26):Spark通讯架构

目录 0. 相关文章链接 1. Spark通信架构概述 2. Spark 通讯架构解析 0. 相关文章链接 Spark文章汇总 1. Spark通信架构概述 Spark 中通信框架的发展&#xff1a; Spark 早期版本中采用 Akka 作为内部通信部件。Spark1.3 中引入 Netty 通信框架&#xff0c;为了解决 Shuf…

靶机渗透之prime1(解法2)

prime1 靶机渗透获取目标shell查看当前用户下的信息查找是否含有有效的备份文件经过查找找到enc文件密码md5格式生成登录靶机Prime提权成功获取flag 靶机渗透 获取目标shell 查看当前用户下的信息 在当前文件夹下&#xff0c;enc文件需要密码&#xff0c;尝试去寻找该文件的密…

移远通信发布新款5G/4G、LPWA和GNSS天线,进一步优化物联网终端性能

2023年7月17日&#xff0c;全球领先的物联网整体解决方案供应商移远通信宣布&#xff0c;再次推出三款新型天线产品&#xff0c;以更优的通信和定位性能&#xff0c;满足各类物联网终端在5G/4G、LPWA和GNSS等技术上的更高设计需求。这三款天线包括&#xff1a; YEMN926J1A&…

【GlobalMapper精品教程】061:快速生成全球任意地区等高线

本文讲解在globalmapper中根据任意指定的区域下载数字地形并生成等高线,支持在CASS中打开并编辑。 文章目录 一、指定下载区域二、下载地形数据三、生成等高线四、投影转换五、导出等高线一、指定下载区域 首先,通过在线地图指定一个区域。当然你有区域范围,可以直接下载。…

爱上PyCharm全新UI的五个理由!让Python开发更个性化

在2023.1版本中&#xff0c; JetBrains官方产品团队对 PyCharm 的外观进行了重新设计&#xff0c;目标是降低视觉复杂性&#xff0c;使用户能够轻松访问基本功能&#xff0c;并根据需要逐级呈现复杂功能 – 打造整洁、现代且专业的外观和质感。 在本文中&#xff0c;我们将进一…

基于python+ResNet50算法实现一个图像分类识别系统入门

一、目录 ResNet50介绍图片模型训练预测项目扩展 在本文中将介绍使用Python语言&#xff0c;基于TensorFlow搭建ResNet50卷积神经网络对四种动物图像数据集进行训练&#xff0c;观察其模型训练效果。 二、ResNet50介绍 ResNet50是一种基于深度卷积神经网络&#xff08;Conv…

ChatGPT助力DevOps的优势与局限

一、前言 DevOps 是一种方法论&#xff0c;旨在提高软件开发和 IT 运营团队的协作和效率。DevOps 涉及各种任务和流程的自动化&#xff0c;例如规划、编码、测试、部署、监控和故障排除。然而&#xff0c;其中一些任务和流程仍然有大量任务需要人工手动处理&#xff0c;而这会…

Debian 系统安装中文输入法-iTOP3588开发板

Debian 系统烧写完成之后&#xff0c;并没有中文输入功能。本文档将介绍如何安装 ibus pinyin 输入法。 首先安装 fcitx 对应的工具&#xff0c;如下图所示&#xff1a; apt-get install fcitx fcitx-tools fcitx-config* fcitx-frontend* fcitx-module* fcitx-ui-* presage …

TDengine 的查询性能与老牌时序数据库相比如何?来看看

在上一篇文章《IoT 场景下写入性能&#xff1a;TDengine16.2 x InfluxDB》中&#xff0c;我们基于 IoT 场景下的 TSBS 时序数据库&#xff08;Time Series Database&#xff09;性能基准测试报告对三大数据库写入性能进行了相关解读&#xff0c;较为直观地展现出了 TDengine 的…

springboot @Async 异步调用接口处理数据

Async 异步背景 新增的数据需要分发给下游业务系统&#xff0c;由于下游业务系统状态未知&#xff0c;所以需要异步发送数据给下游业务系统。 系统生效按钮--->controller新增-->异步调用servcie--->数据集成 在springboot框架中实现步骤 首先在启动类上加上Enable…

数据分析:扩展企业微信、钉钉、飞书等告警通知渠道

本章节主要讲述如何扩展告警的通知渠道&#xff0c;以便将告警发送到第三方应用中 企业微信 实现目标 ●在鸿鹄中创建的告警被触发后&#xff0c;将告警通知发送至指定的企业微信群聊 配置步骤 1、打开“企业微信”&#xff0c;点击告警群右上角的“...”按钮 2、点击“群机器人…

opencv双目视觉标定、匹配和测量

双目视觉原理方面参照《学习Opencv》和大牛博客 http://blog.csdn.net/chenyusiyuan/article/details/5970799中16-19系列博客。本文主要记录我自己在双目视觉标定,立体匹配,测量中遇到的问题和解决方法,并附有代码,文末有代码下载的地址,欢迎交流。 博主使用的相机是USB双…