【JavaEE精炼宝库】多线程1(认识线程 | 创建线程 | Thread 类)

目录

一、认识线程

1.1 线程的概念:

1.2 为什么需要线程:

1.3 面试题.谈谈进程和线程的区别:

1.4 Java的线程和操作系统线程的关系:

二、创建线程

2.1 创建线程的5种写法:

2.1.1 写法1.继承 Thread 类:

2.1.2 写法2.实现 Runnable 接口:

2.1.3 写法3.继承 Thread 使用匿名内部类:

2.1.4 写法4.实现 Runnable 使用匿名内部类:

2.1.5 写法5.使用lambda(推荐写法):

2.2 run方法和start方法的区别:

2.3 多线程的优势:

三、Thread类及常见方法

3.1 Thread 的常见构造方法:

3.2 jconsole使用过程:

3.3 Thread的常见属性:

3.3.1 属性列表:

3.3.2 前后台线程的关系:


一、认识线程

1.1 线程的概念:

一个线程就是一个 "执行流" 。每个线程之间都可以按照顺序执行自己的代码。多个线程之间 "同时" 执行着多份代码。

1.2 为什么需要线程:

(1)首先经过多年的发展,“并发编程” 已成成为 “刚需”。

• 单核CPU的发展遇到了瓶颈.要想提高算力,就需要多核CPU.而并发编程能更充分利用多核CPU资源。

 • 有些任务场景需要 "等待IO",为了让等待IO的时间能够去做一些其他的工作,也需要用到并发编程。

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

• 创建线程比创建进程更快。

• 销毁线程比销毁进程更快。

• 调度线程比调度进程更快。

(3)最后线程虽然比进程轻量,但是人们还不满足,于是又有了 "线程池" (ThreadPool)和 "协程"  (Coroutine)。

本文章主要介绍多线程,有关线程池和协程的概念后续会单独再写文章解释。

1.3 面试题.谈谈进程和线程的区别:

主要有如下四点:

• 进程是包含线程的。每个进程至少有⼀个线程存在,即主线程。

• 进程和进程之间不共享内存空间。同⼀个进程的线程之间共享同⼀个内存空间。

• 进程是系统分配资源的最小单位,线程是系统调度的最小单位。

• 一个进程挂了一般不会影响到其他进程。但是一个线程挂了,可能把同进程内的其他线程一起带走(整个进程崩溃)。

1.4 Java的线程和操作系统线程的关系:

线程是操作系统中的概念。操作系统内核实现了线程这样的机制,并且对用户层提供了一些API供用户使用(例如Linux的pthread库)。

Java标准库中Thread类可以视为是对操作系统提供的API进行了进一步的抽象和封装。

二、创建线程

下面我会提供创建线程的常见的5中写法,希望友友们都要掌握。下面经常会用到run方法和start方法,关于它们的区别,大家可以先把这5中写法看完后,我在后面有写区别🤩🤩🤩。

2.1 创建线程的5种写法:

2.1.1 写法1.继承 Thread 类:

继承 Thread 来创建⼀个线程类。写法如下:

class MyThread extends Thread{@Overridepublic void run() {System.out.println("hello Thread");System.out.println("Thread end");}
}
public class demo1 {public static void main(String[] args) {Thread t = new MyThread();//向上转型t.start();//启动线程}
}

这里解释一下为什么不能直接直接创建一个Thread对象,而是要再写一个Thread的子类:这是因为我们要重写 run 方法,如果不重写,直接调用原生的,会达不到我们的预期,这显然不是我们想看到的。

运行结果:

2.1.2 写法2.实现 Runnable 接口:

• Runnable接口源码:

通过观察其源码我们不难发现这是一个 “函数式接口” ,所以我们后面有一种写法就会利用到lambda表达式,里面涉及到的一些 “变量捕获” 的知识如果友友忘了的话要记得复习呀。

这个相比于第一个写法的好处是:能够起到解耦合的作用,例如当前是通过多线程的方式执行的,未来也可以很方便改成基于线程池的方式执行,也可以改成基于虚拟线程的方式执行(改动成本比较小),而继承Thread的写法基本就只适用于多线程。具体写法如下:

class MyRunnable implements Runnable{@Overridepublic void run() {System.out.println("hello Thread");System.out.println("Thread end");}
}
public class demo2 {public static void main(String[] args) {Thread t = new Thread(new MyRunnable());t.start();}
}

运行结果: 

2.1.3 写法3.继承 Thread 使用匿名内部类:

这个写法的效果和 2.1 的写法效果没有任何区别,因为使用匿名内部类本来就是为了方便。具体写法如下:

public class demo3 {public static void main(String[] args) {Thread t = new Thread(){@Overridepublic void run() {System.out.println("hello Thread");System.out.println("Thread end");}};t.start();}
}

运行结果和前面一样就不贴了。

2.1.4 写法4.实现 Runnable 使用匿名内部类:

和写法2效果一样:

public class demo4 {public static void main(String[] args) {Thread t = new Thread(new MyRunnable(){@Overridepublic void run() {System.out.println("hello Thread");System.out.println("Thread end");}});t.start();}
}

2.1.5 写法5.使用lambda(推荐写法):

这个是比较推荐的写法,因为是 “函数式接口” 我们就可以使用lambda表达式来简化写法。具体写法如下:

public class demo5 {public static void main(String[] args) {Thread t = new Thread(()->{System.out.println("hello Thread");System.out.println("Thread end");});t.start();}
}

 上述 5 种写法本质都是要把线程执行的任务内容表示出来,通过 Thread 的 start 来创建 / 启动系统中的线程。Thread 对象和操作系统内核中的线程是一一对应的关系。 

2.2 run方法和start方法的区别:

之前我们已经看到了如何通过覆写 run 方法创建一个线程对象,但线程对象被创建出来并不意味着线程就开始运行了。

• 覆写 run 方法是提供给线程要做的事情的指令清单。

• 线程对象可以认为是把 李四、王五叫过来了。

• 而调用 start() 方法,就是喊⼀声:”行动起来!“,线程才真正独立去执行了。

总而言之:调用 start 方法,才真的在操作系统的底层创建出一个线程。

2.3 多线程的优势:

利用多线程在一些场合可以提高程序的运行速度。具体案例如下:

前置知识:

• 使用 System.nanoTime() 可以记录当前系统的 纳秒 级时间戳。

• serial 串行的完成一系列运算。concurrency 使用两个线程并行的完成同样的运算。

如果对串行和并行不了解的话可以前往:JavaEE前置知识 中查看并行与并发的区别。

public class ThreadAdvantage {// 多线程并不⼀定就能提⾼速度,可以观察,count 不同,实际的运⾏效果也是不同的private static final long count = 10_0000_0000;public static void main(String[] args) throws InterruptedException {// 使⽤并发⽅式concurrency();// 使⽤串⾏⽅式serial();}private static void concurrency() throws InterruptedException {long begin = System.nanoTime();// 利⽤⼀个线程计算 a 的值Thread thread = new Thread(new Runnable() {@Overridepublic void run() {int a = 0;for (long i = 0; i < count; i++) {a--;}}});thread.start();// 主线程内计算 b 的值int b = 0;for (long i = 0; i < count; i++) {b--;}// 等待 thread 线程运⾏结束thread.join();// 统计耗时long end = System.nanoTime();double ms = (end - begin) * 1.0 / 1000 / 1000;System.out.printf("并发: %f 毫秒%n", ms);}private static void serial() {// 全部在主线程内计算 a、b 的值long begin = System.nanoTime();int a = 0;for (long i = 0; i < count; i++) {a--;}int b = 0;for (long i = 0; i < count; i++) {b--;}long end = System.nanoTime();double ms = (end - begin) * 1.0 / 1000 / 1000;System.out.printf("串⾏: %f 毫秒%n", ms);}
}

案例结果如下:

可以看到速度快了两倍多。不使用多线程的并发就会出现 “一核有难,多核围观” 的现象。

三、Thread类及常见方法

Thread 类是 JVM 用来管理线程的一个类,换句话说,每个线程都有一个唯一的 Thread 对象与之关联。

用我们上面的例子来看,每个执行流,也需要有一个对象来描述,类似下图所示,而 Thread 类的对象就是用来描述一个线程执行流的,JVM 会将这些 Thread 对象组织起来,用于线程调度,线程管理。

3.1 Thread 的常见构造方法:

我们最常使用的是第三个,至于第五个目前在实际开发中更多的是被线程池取代了,这里只演示第三个。演示如下:

public class demo6 {public static void main(String[] args) {Thread t = new Thread(()->{while(true){System.out.println("hello Thread");try {Thread.sleep(1000);} catch (InterruptedException e) {throw new RuntimeException(e);}}},"我的Thread");t.start();while(true){System.out.println("hello Main");try {Thread.sleep(1000);} catch (InterruptedException e) {throw new RuntimeException(e);}}}
}

我们可以利用使用 jconsole 命令观察线程。

3.2 jconsole使用过程:

我们利用上面的程序来进行连接,连接的时候要保证程序在运行中。

• 打开 c 盘并进入 Program Files :

• 进入 Java 中的 jdk :

• 进入 bin 后找到 jconsole 后以管理员的身份运行它:

•  看到这个就成功找到 jconsole 了:

记得连接我们运行的java程序。

注意:程序一定要保证在运行状态,比如我们写一个while(true)循环来保证我们连接的时候程序在跑,不然我们是连接不到的。

• 查看结果:

点击线程,在下面我们能看到main和我的Thread(修改命名)。

完成上面步骤我们已经成功利用 jconsole 查看运行的 java 线程。

3.3 Thread的常见属性:

3.3.1 属性列表:

属性解释:

• ID 是线程的唯一标识,不同线程不会重复。

• 名称是各种调试工具用到。

• 状态表示线程当前所处的一个情况。

• 优先级高的线程理论上来说更容易被调度到。

• 关于后台线程,需要记住一点:JVM会在一个进程的所有非后台线程结束后,才会结束运行。

• 是否存活,即简单的理解,为 run 方法是否运行结束了。

• 线程的中断问题,下面会单独讲。

3.3.2 前后台线程的关系:

• 前台线程:前台线程如果不运行结束的话,此时 Java 进程是一定不会结束的。

• 后台线程:后台线程即使继续在执行,也不能阻止 Java 进程结束。

我们默认创建的线程都是前台线程。我们可以利用 setDaemon 方法来把线程设置为后台线程。

注意:关于线程的各种属性的设置,都要放在 start 之前,一旦线程已经启动了,那么开弓就没有回头箭,这个时候再设置就来不及了,还会返回一个异常。

测试案例:

public class demo7 {public static void main(String[] args) {Thread t = new Thread(()->{while(true){System.out.println("hello Thread");try {Thread.sleep(1000);} catch (InterruptedException e) {throw new RuntimeException(e);}}});t.setDaemon(true);//把t设置为后台线程t.start();//启动线程try {Thread.sleep(1000);} catch (InterruptedException e) {throw new RuntimeException(e);}System.out.println("Main end");}
}

友友们可以把这个代码贴到自己的编译器上面,看看有没有 setDaemon 的区别。

案例效果如下:

结语:

其实写博客不仅仅是为了教大家,同时这也有利于我巩固知识点,和做一个学习的总结,由于作者水平有限,对文章有任何问题还请指出,非常感谢。如果大家有所收获的话还请不要吝啬你们的点赞收藏和关注,这可以激励我写出更加优秀的文章。

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

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

相关文章

【redis】Redis五种常用数据类型和内部编码,以及对String字符串类型的总结

˃͈꒵˂͈꒱ write in front ꒰˃͈꒵˂͈꒱ ʕ̯•͡˔•̯᷅ʔ大家好&#xff0c;我是xiaoxie.希望你看完之后,有不足之处请多多谅解&#xff0c;让我们一起共同进步૮₍❀ᴗ͈ . ᴗ͈ აxiaoxieʕ̯•͡˔•̯᷅ʔ—CSDN博客 本文由xiaoxieʕ̯•͡˔•̯᷅ʔ 原创 CSDN 如…

C++ | Leetcode C++题解之第80题删除有序数组中的重复项II

题目&#xff1a; 题解&#xff1a; class Solution { public:int removeDuplicates(vector<int>& nums) {int n nums.size();if (n < 2) {return n;}int slow 2, fast 2;while (fast < n) {if (nums[slow - 2] ! nums[fast]) {nums[slow] nums[fast];slo…

【doghead】mac与wsl2联通

mbp 设置为发送端,那么要能与windows上 wsl2的ubutnu通信。 mbp的 uv 构建ok zhangbin@zhangbin-mbp-2  ~/tet/Fargo/zhb-bifrost/Bifrost-202403/worker/third_party/libuv   main clion使用lldb cmake构建 更新git2.45.0啊

盘点自动驾驶的技术发展趋势

自动驾驶技术在不断发展变快&#xff0c;我们之前提过算法岗如今越来越卷&#xff0c;从今年的就业局势看&#xff0c;前年还属于蓝海行业的自动驾驶&#xff0c;今年就已经满满关上了招揽之门——呈红海之势。作为在这个行业中摸爬滚打的一以子&#xff0c;我们到底该如何纵观…

会员卡积分收银源码系统 支持多门店系统 带完整的安装代码包以及安装搭建教程

在数字化浪潮的推动下&#xff0c;传统零售行业面临着巨大的转型压力。为了满足现代消费者多样化的需求&#xff0c;提高门店管理效率和顾客满意度&#xff0c;小编给大家分享一款会员卡积分收银源码系统——支持多门店系统&#xff0c;并附带了完整的安装代码包以及安装搭建教…

君正T31移植电源IC—CW2015芯片简介

CW2015芯片简介 CW2015 是一款超紧凑、低成本、主机侧/电池组侧、无传感电阻器的电量计量系统 IC&#xff0c;适用于手持和便携式设备中的锂离子 (Li) 电池。CW2015 包括一个14 位Sigma-Delta ADC、一个精密电压基准和内置准确温度传感器。该 IC 允许最终用户消除占用大量电路板…

【Vue3进阶】- Pinia

什么是Pinia Pinia 是 Vue 的专属状态管理库&#xff0c;它允许你跨组件或页面共享状态。它提供了类似于 Vuex 的功能&#xff0c;但比 Vuex 更加简单和直观。 需要在多个组件之间共享状态或数据时使用 Pinia 的 store&#xff0c;这样可以避免 props 和 eventBus 等传统方法…

【stm32笔记】DSP库调用

参考&#xff1a;DSP库调用 , __CC_ARM,__TARGET_FPU_VFP, __FPU_PRESENT1U, ARM_MATH_CM4 ,USE_HAL_DRIVER,STM32F407xx,ARM_MATH_CM4,__FPU_USED1U,__FPU_PRESENT1U,__CC_ARM,ARM_MATH_MATRIX_CHECK,ARM_MATH_ROUNDING把需要的库复制出来单独用&#xff0c;方便移植

KEIL declaration may not appear after executable statement in block

KEIL declaration may not appear after executable statement in block 这个问题也是比较经典&#xff0c;就是不允许你的变量定义位置不允许在下边的代码区域&#xff0c;只允许在最上方 ‍ 修改编码模式为C99解决 ‍ ​​

(Java)心得:LeetCode——15.三数之和

一、原题 给你一个整数数组 nums &#xff0c;判断是否存在三元组 [nums[i], nums[j], nums[k]] 满足 i ! j、i ! k 且 j ! k &#xff0c;同时还满足 nums[i] nums[j] nums[k] 0 。请你返回所有和为 0 且不重复的三元组。 注意&#xff1a;答案中不可以包含重复的三元组。…

AI大模型探索之路-训练篇15:大语言模型预训练之全量参数微调

系列篇章&#x1f4a5; AI大模型探索之路-训练篇1&#xff1a;大语言模型微调基础认知 AI大模型探索之路-训练篇2&#xff1a;大语言模型预训练基础认知 AI大模型探索之路-训练篇3&#xff1a;大语言模型全景解读 AI大模型探索之路-训练篇4&#xff1a;大语言模型训练数据集概…

奇诡 matlab 小 bug matlab git需要记录的改动太多

似乎是我有一次添加了太多的路径之后的事情。但是不敢说一定是这个导致的&#xff1a; 症状&#xff1a;只要对文本进行任何编辑操作&#xff0c;工作区就会出现"Processing … Cancel"的提示&#xff0c;如果不管的话这个提示不会消失&#xff0c;同时matlab变得越来…

9.spring-图书管理系统

文章目录 1.开发项目流程1.1开发开发1.2数据库的设计 2.MySQL数据库相关代码3.构造图书结构3.1用户登录3.2图书列表3.3图书添加3.4图书删除3.4.1批量删除 3.5图书查询(翻页) 4.页面展示4.1登录页面4.2列表页面4.3增加图书页面4.4修改图书信息页面 5.功能展示5.1增加图书信息5.2…

语言基础 /CC++ 可变参函数设计与实践,va_ 系列实战详解(强制参数和变参数的参数类型陷阱)

文章目录 概述va_ 系列定义va_list 类型va_start 宏从变参函数的强制参数谈起宏 va_start 对 char 和 short 类型编译告警宏 va_start 源码分析猜测 __va_start 函数实现 va_arg 宏宏 va_arg 无法接受 char 和 short为啥va_arg可解析int却不能解析float类型&#xff1f;宏 va_a…

基于单片机的直流电机测速装置研究与设计

摘要: 基于单片机的直流电机测速装置采用了对直流电机的中枢供电回路串联取样电阻的方式实现对电机转速的精确实时测量。系统由滤波电路、信号放大电路、单片机控制电路以及稳压电源等功能模块电路构成。工作过程中高频磁环作为载体&#xff0c;利用电磁感应的基本原理对直流电…

spring boot 基础案例【4】使用Swagger2构建强大的API文档

教程1 案例教程 案例仓库 在线编程 在线编辑器运行&#xff1a;mvn spring-boot:run 教程2 基础教程 教程仓库 在线编程 本案例所在的仓库 本案例所在的文档 进入正文 1.文件目录 2.应用主类 地址:2.x/chapter2-2/src/main/java/com/didispace/chapter22/Chapter22Applicatio…

Secnet-智能路由系统 actpt_5g.data 信息泄露漏洞复现

0x01 产品简介 Secnet安网智能AC管理系统是广州安网通信技术有限公司&#xff08;简称“安网通信”&#xff09;的无线AP管理系统。 0x02 漏洞概述 Secnet-智能路由系统 actpt_5g.data 接口存在信息泄露漏洞&#xff0c;未经身份验证的远程攻击者可以利用此漏洞获取系统账户…

【JVM基础篇】JVM入门介绍

JVM入门介绍 为什么学习JVM 岗位要求 解决工作中遇到的问题 性能调优 真实案例 导出超大文件&#xff0c;系统崩溃从数据库中查询超大量数据出错消费者消费来不及导致系统崩溃Mq消息队列接受消息导致的内存泄漏业务高峰期系统失去响应 初识JVM 什么是JVM&#xff1f; JV…

C 语言中怎么产生真正的随机数?

在C语言中&#xff0c;要产生真正的随机数&#xff0c;我们通常使用标准库中的 <stdlib.h> 头文件中提供的随机数生成函数。 这些函数可以生成伪随机数&#xff0c;但它们在一定程度上是随机的&#xff0c;足以满足大多数应用程序的需求。 1. 伪随机数生成函数 C标准库…