【后端开发】JavaEE初阶—Theard类及常见方法—线程的操作(超详解)

前言:

🌟🌟本期讲解多线程的知识哟~~~,希望能帮到屏幕前的你。

🌈上期博客在这里:【后端开发】JavaEE初阶—线程的理解和编程实现-CSDN博客

🌈感兴趣的小伙伴看一看小编主页:GGBondlctrl-CSDN博客

目录

📚️1.引言

📚️2.Thread常见的构造方法

2.1创建对象名字

1.第一种创建线程对象,并命名

2.使用Runnable对象,并命名

3.检查线程名字

 2.2线程组

 📚️3.Theard类常见属性

3.1属性ID与名称

3.2是否后台线程

 3.3是否存活

 📚️4.启动线程

4.1多次启动线程

 4.2实现多次启动线程

4.3start()与run()使用

📚️5.终止一个线程

5.1设置标志位isQuit

5.2标志位的替代

 📚️6.等待线程

📚️7.获取线程引用

7.1使用Thread继承

7.2不使用Thread继承 

 📚️8.总结

 

📚️1.引言

 Hello!!! 小伙伴们咱们又见面啦,继小编上期讲解了多线程编程的相关重要知识,以及如何实现线程的创建之后,本期将继续讲解关于多线程的相关方法和属性,开始发车了加油加油~~~🥳🥳🥳

且听小编讲解,包你学会!!!

📚️2.Thread常见的构造方法

主要方法如下:

这里的创建线程对象,以及使用Runnable对象创建线程小编是在上期是讲解过的,想要了解的小伙伴可以去小编主页看看;那么接下来就来几个陌生的方法吧

2.1创建对象名字

1.第一种创建线程对象,并命名

代码如下:

Thread t=new Thread(()->{System.out.println("这是thread线程");try {Thread.sleep(1000);} catch (InterruptedException e) {throw new RuntimeException(e);}},"我的线程");

 注意:小编这里使用了lambda表达式进行演示,其中在方法体内进行了补充,在方法体后,用双引号表示的就是为这个线程所编写的名字;

2.使用Runnable对象,并命名

代码如下:

Thread t1=new Thread(new Runnable() {@Overridepublic void run() {}},"这是我的线程");

注意:此时小编没有对run方法进行重写,但是仍然对线程进行了命名;

3.检查线程名字

这里就要用到JDK中的jconsole了,这个在JDK中存在;

然后点击进入jconsole,并进行我们创建的线程的连接,图片展示:

很明显这里小编创建的项目是threadDemo6,并点击连接,找到线程,就可以发现我们创建线程的名字了,图片展示如下:

这就是我们创建线程的名字啦~~~

注意:在执行上述操作时,一定要保证从开始到结束中,idea上的线程代码要跑起来,否则在连接处就无法找到我们创建的线程; 

当然这里也可以使用getName()方法,小编会在后面演示~~~

 2.2线程组

这里的线程组是java中的概念,和系统内核中的线程组是不一样的;

 线程组的作用:

1. 组织线程:可以将多个线程归为一个线程组,方便对相关线程进行整体操作和管理。
2. 控制权限:线程组可以控制其包含的线程的访问权限,例如设置是否允许某个线程组中的线程修改系统资源等。
3. 异常处理:当一个线程组中的某个线程抛出未捕获的异常时,线程组可以对这个异常进行统一的处理。

这里的线程组,咱们了解即可~~~

 📚️3.Theard类常见属性

常见属性如下:

这里的getState()方法就是描述线程的状态,进程存在就绪状态与阻塞状态,那么线程也存在对应的状态,以及这里的getPriority()方法描述的是线程的调度优先级,但是效果不明显,至于收否被中断,后面小编会进行讲解;接下来即小编重点实例讲解的了;

3.1属性ID与名称

如何获取线程的ID与名称;代码如下:

Thread t=new Thread(()->{System.out.println("这是thread线程");try {Thread.sleep(1000);} catch (InterruptedException e) {throw new RuntimeException(e);}},"我的线程");
t.start();System.out.println(t.getId()+"  "+t.getName());

注意:在通过Theard实例调用线程的ID和名字时,对应的是JVM自动分配的身份标识,对于名字一般是Thread-0,Thread-1......但是自主给线程起名后,对应的名字也会变为所起名字;

3.2是否后台线程

按照之前的理解,当main函数执行结束后,整个函数就应该执行结束了,但是在多线程中并不是如此,因为JVM内置线程为后台线程,而后台线程不会阻止线程结束

我们创建的代码线程默认为前台线程,前台线程会阻止线程结束,所以即使main函数执行完了,那么线程仍然不会结束此时;

此时我们就要将前台线程改为后台线程,代码如下:

 public static void main(String[] args) {Thread t = new Thread(new Runnable() {@Overridepublic void run() {while (true) {System.out.println("hello thread");}}}, "这是我的线程");// 在 start 之前, 设置线程为 后台线程 (不能在 start 之后设置)t.setDaemon(true);t.start();}

此时主线程走完了就不会有任何输出:

注意:前台线程会阻止进程结束,而后台线程不会阻止进程结束,并且在改写前台线程为后台线程时,必须在创建线程之前!!!

 3.3是否存活

这里的是否存活表示的是内核中的PCB线程是否存在;

Thread实例,虽然来说表示的是一个线程,但是这里和内核中的PCB线程的生命周期是不一样的

代码实例如下:

 Thread t=new Thread(()->{System.out.println("这是thread线程");try {Thread.sleep(1000);} catch (InterruptedException e) {throw new RuntimeException(e);}},"我的线程");System.out.println("线程创建之前"+t.isAlive());//falset.start();System.out.println("线程启动之后"+t.isAlive());//trueThread.sleep(3000);//保证t线程结束System.out.println("t线程结束之后"+t.isAlive());//false

这里的输出结果就如同上述注解;

注意:内核中的PCB线程是在t调用start方法后创建线程才会存在,当线程中run()方法执行完后,内核PCB被释放了,此时isAlive表示为没有存活;

这里的主线程休眠是为了保证另一线程执行完毕;

 📚️4.启动线程

对于启动线程来说就使用start方法就行了,但是这里要进行扩展

4.1多次启动线程

代码如下:

Thread t=new Thread(()->{System.out.println("这是thread线程");},"我的线程");t.start();t.start();

这里小编进行了线程的两次启动,那么结果就是:

很明显这里抛出异常了即:非法的线程状态异常~~~

注意:线程只能启动一次,一个线程实例对象,多次调用start方法就会抛出非法线程状态异常

 4.2实现多次启动线程

由于上述讲到,一个线程实例化对象不能多次启动线程,那么对于多次启动线程,就要实现多个线程,多个启动

代码实例如下:

public class ThreadDemo10 {public static void main(String[] args) {Thread t1 = new Thread(() -> {System.out.println("hello1");});Thread t2 = new Thread(() -> {System.out.println("hello2");});t1.start();t2.start();}

此时我们创建了两个线程,这里就可以实现多次启动线程啦~~~

4.3start()与run()使用

对于这两种方法,两者其实是互不相干的;

start()方法:是通过系统调用API实现了一个线程的创建,其中的JVM在创建好线程后自动调用run方法,此时就有多个线程;

run()方法:就是之前在JavaSE部分中,类中方法的调用一致,但是这里调用后,没有创建新的现场,仍然为单线程;

代码实例:

class MyThread4 extends Thread {@Overridepublic void run() {while (true) {System.out.println("hello thread");}}
}public class ThreadDemo11 {public static void main(String[] args) {Thread t = new MyThread4();t.start();// t.run();while (true) {System.out.println("hello main");}}

注意:在如上述代码中,当调用start方法后,会创建两个线程,那么会不断输出(hello thread)和(hello main)这两个输出交替出现,但是调用run方法,就会陷入死循环打印(hello thread)不会向下执行主线程了;

📚️5.终止一个线程

这里终止一个线程就是:让线程run方法执行完毕;

5.1设置标志位isQuit

代码如下:

 public static boolean isQuit=true;public static void main(String[] args) throws InterruptedException {Thread t=new Thread(()->{while (isQuit){System.out.println("这是一个thread线程,在工作");try {Thread.sleep(2000);}catch (InterruptedException e){throw new RuntimeException();}}System.out.println("线程结束");});t.start();Thread.sleep(4000);System.out.println("让t线程结束");isQuit=false;

此时的输出结果就是:

此时我们想让线程结束,那么就要将循环内的判断条件进行改变,即将isQuit的值变为false,并且为了观察,进行必要的输出打印;

 那么我们将代码中的isQuit改为主线程的局部变量,就是不可行的!!!

因为lambda表达式/匿名内部类存在变量捕获的问题,此时这里的isQuit已经被final修饰了,并且匿名内部类是可以访问外部类的成员变量的,所以不可以改为main函数的局部变量;

5.2标志位的替代

这里就用Thread.currentThread().isInterrupted()来代替isQuit,但是这里表示为false,interrupt表示修改值

代码如下:

 public static void main(String[] args) throws InterruptedException {Thread t=new Thread(()->{while (!Thread.currentThread().isInterrupted()){System.out.println("线程在工作");try {Thread.sleep(2000);} catch (InterruptedException e) {e.printStackTrace();}}System.out.println("工作结束");});t.start();Thread.sleep(4000);System.out.println("结束线程工作");t.interrupt();//类似改变标志的布尔值}

此时我进行结果打印时,发现抛出异常了:

当我们输出结束线程工作时,这里没有结束继续工作,并抛出异常:休眠终止异常~~~

注意:

在执行 sleep时调用interrupt会导致sleep提前唤醒;会导致抛出上述异常,或者将Thread实例中的isInterrupted()标志位,改为false,这就会导致程序继续执行;

解决方法 :

此时我们就可以加上break,让线程立即结束,代码如下:

 while (!Thread.currentThread().isInterrupted()){System.out.println("线程在工作");try {Thread.sleep(2000);} catch (InterruptedException e) {e.printStackTrace();// break;}}

当然或者直接省去sleep休眠状态,直接一直工作知道终止线程;

经过以上解释,程序的终止是一种软性操作,需要线程的配合才能实现!!!

 📚️6.等待线程

由于线程是调度执行的,底层调度是不确定的,但是可以通过一些API来影响线程的执行顺序,此时join就提供了这样的操作;

代码如下:

 public static void main(String[] args) throws InterruptedException {Thread t = new Thread(() -> {int n = 5;for (int i = 0; i < n; i++) {System.out.println("我是一个线程, 正在工作中...");try {Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}}System.out.println("线程执行结束");});t.start();// 这个操作就是线程等待.t.join();System.out.println("这是主线程, 期望这个日志在 t 结束后打印");}

 那么此时小编就使用了join来等待线程的执行完毕,再输出主线程,输出结果:

那么此时达到了我们的要求;但是我们运用sleep方法也可以实现,为啥不用它呢???

解答:在使用sleep方法时要知道线程结束的具体时间,因为休眠时间是要根据线程执行完毕才能够实现的,而使用join就不必知道这一点;

注意:

在使用join方法时要进行异常的抛出;

在那个线程使用join方法,谁就处于等待状态,而调用的线程实例就是被等待的一方;

使用sleep方法可以实现上述需求,但是需要知道线程的执行时间,所以不推荐;

join中也可以添加等待时间,如果不添加就是“死等”状态,一般在计算机中不推荐死等

 这里也可以使用线程等待来实现一系列运算;

public static long sum=0;public static void main(String[] args)throws InterruptedException {Thread t1=new Thread(()->{for (long i = 0; i <50 ; i++) {sum+=i;}});Thread t2=new Thread(()->{for (long i = 50; i <=100 ; i++) {sum+=i;}});t1.start();t2.start();long begin=System.currentTimeMillis();t1.join();t2.join();long end=System.currentTimeMillis();System.out.println("sum="+sum);System.out.println("等待时间"+(end-begin)+"ms");}

这里小编就使用多线程,和等待机制来实现算数相加,此时在两个线程开启后,在让主线程进行等待,这里使用时间打印可以来表示单个线程和多个线程来执行这个任务的时间相差,小编这里就不再过多演示;

📚️7.获取线程引用

7.1使用Thread继承

代码如下:

 public static void main(String[] args) throws InterruptedException {Thread t=new mythread7();t.start();   }
}
class mythread7 extends Thread{@Overridepublic void run(){System.out.println(this.getId()+" "+this.getName());}
}

注意:在Thread继承中可以使用this拿到线程实例;

7.2不使用Thread继承 

代码如下:

 public static void main(String[] args) throws InterruptedException {Thread t=new Thread(()->{Thread t1=Thread.currentThread();System.out.println(t1.getId());});t.start();}

注意:如果是Runnable或者是lambda表达式,this就无法指向Thread对象,这个时候就要用到Thread.currentThread()

至于为啥不直接使用(t)来得到 属性:

在 Lambda 表达式中,变量必须是最终变量或有效最终变量,而 t 是一个非最终变量,因为他还没有完全初始化

 📚️8.总结

💬💬小编本期讲解了关于线程的某些重要属性和方法,例如线程的启动,终止,等待以及常见的属性的获取方法和构造方法,并附上代码供小伙伴们参考~~~

代码在这里哟:GGBondlctrl/Thread (gitee.com)😊 😊 😊

🌅🌅🌅~~~~最后希望与诸君共勉,共同进步!!!


💪💪💪以上就是本期内容了, 感兴趣的话,就关注小编吧。

                                                                 😊😊  期待你的关注~~~

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

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

相关文章

计算机毕业设计之:基于深度学习的路面检测系统(源码+部署文档+讲解)

博主介绍&#xff1a; ✌我是阿龙&#xff0c;一名专注于Java技术领域的程序员&#xff0c;全网拥有10W粉丝。作为CSDN特邀作者、博客专家、新星计划导师&#xff0c;我在计算机毕业设计开发方面积累了丰富的经验。同时&#xff0c;我也是掘金、华为云、阿里云、InfoQ等平台…

鸿蒙OpenHarmony【轻量系统内核扩展组件(CPU占用率)】子系统开发

基本概念 CPU&#xff08;中央处理器&#xff0c;Central Processing Unit&#xff09;占用率分为系统CPU占用率和任务CPU占用率。 系统CPU占用率&#xff1a;是指周期时间内系统的CPU占用率&#xff0c;用于表示系统一段时间内的闲忙程度&#xff0c;也表示CPU的负载情况。系…

INIT与init_array

INIT与init array 1.so执行JNI_OnLoad之前&#xff0c;还会执行俩个构造函数init 和init array 在so加载时候有这个过程&#xff1a; .init -> .init array -> JNI_Onload -> java_com_xxx 在脱壳的过程中会在一些系统级的.so中下断点比如&#xff1a;fopen&#x…

GUI编程19:贪吃蛇小游戏及GUI总结

视频链接&#xff1a;21、贪吃蛇之界面绘制_哔哩哔哩_bilibilihttps://www.bilibili.com/video/BV1DJ411B75F?p21&vd_sourceb5775c3a4ea16a5306db9c7c1c1486b5 1.游戏中用的的图片素材 1.贪吃蛇游戏的主启动类StartGame&#xff1b; package com.yundait.snake;import j…

缓存的思考与总结

缓存的思考与总结 什么是缓存缓存命中率数据一致性旁路模式 Cache aside双写模式直写模式 write through异步写 Write Behind 旁路和双写 案例 新技术或中间的引入&#xff0c;一定是解决了亟待解决的问题或是显著提升了系统性能&#xff0c;并且这种改变所带来的增幅&#xff…

【开源服务框架】Dubbo

&#x1f384;欢迎来到边境矢梦的csdn博文&#x1f384; &#x1f384;本文主要梳理Java面试中开源服务框架Dubbo会涉及到的知识点 &#x1f384; &#x1f308;我是边境矢梦&#xff0c;一个正在为秋招和算法竞赛做准备的学生&#x1f308; &#x1f386;喜欢的朋友可以关注一…

GAMES101(15节)

Irradiance辐射度量学 辐射度量学在渲染领域&#xff0c;可以帮助理解基于物理的光照模型 radiant energy辐射能量Q&#xff0c;累计总能量&#xff08;单位J joule焦耳&#xff09;&#xff0c;就像太阳能板&#xff0c;光照时间越长接收能量越多&#xff0c;收到的能量总和…

jetlinks物联网平台学习2(加盐算法登陆)

加盐算法 加盐算法加密验证密码是否正确 对于传统的MD5加密&#xff0c;比更传统的直接保存账号密码稍微安全一点。 md5加密是一种hash算法 比如对于123456来说&#xff0c;md5算法结果一定是e10adc3949ba59abbe56e057f20f883e 这个结果是固定的。于是有的人准备一张彩虹表 预先…

ECharts基础使用方法 ---vue

1.安装依赖文件 仔细看项目" README.md " 描述&#xff0c;确定用什么安装 npm npm install echarts --save //官网推荐使用 pnpm pnpm install echarts --save 其他也是 在项目根目录&#xff0c;打开当前目录命令控制栏&#xff0c;输入以上命令并运行 安装成功后…

第十三章:使用html和css做一个静态登录网页练习

我们在使用浏览器 浏览某些网站的时候 有可能会遇到登录这种网页,这种网页是怎么制作出来的呢? 下面 我就来分享一个简单的 登录页 实现方案! 登录页面的作用: 身份验证:登录页面的核心作用就是验证用户身份。用户输入用户名(或邮箱、手机号)和密码,系统通过验证来判断…

[数据结构]无头单向非循环链表的实现与应用

文章目录 一、引言二、线性表的基本概念1、线性表是什么2、链表与顺序表的区别3、无头单向非循环链表 三、无头单向非循环链表的实现1、结构体定义2、初始化3、销毁4、显示5、增删查改 四、分析无头单向非循环链表1、存储方式2、优点3、缺点 五、总结1、练习题2、源代码 一、引…

尚品汇-秒杀商品定时任务存入缓存、Redis发布订阅实现状态位(五十一)

目录&#xff1a; &#xff08;1&#xff09;秒杀业务分析 &#xff08;2&#xff09;搭建秒杀模块 &#xff08;3&#xff09;秒杀商品导入缓存 &#xff08;4&#xff09;redis发布与订阅实现 &#xff08;1&#xff09;秒杀业务分析 需求分析 所谓“秒杀”&#xff0…

百度智能云API调用

植物识别API import base64 import urllib import requestsAPI_KEY = "你的图像识别API_KEY" SECRET_KEY = "你的图像识别SECRET_KEY"def main():url = "https://aip.baidubce.com/rest/2.0/image-classify/v1/plant?access_token=" + get_acc…

12、等保安全通用要求

数据来源&#xff1a;12.等保安全通用要求_哔哩哔哩_bilibili 基本要求

docker启动mysql未读取my.cnf配置文件问题

描述 在做mysql主从复制配置两台mysql时&#xff0c;从节点的my.cnf配置为&#xff1a; [mysqld] datadir /usr/local/mysql/slave1/data character-set-server utf8 lower-case-table-names 1 # 主从复制-从机配置# 从服务器唯一 ID server-id 2 # 启用中继日志 relay-l…

thop计算模型复杂度(params,flops)

thop安装 -pip install thop在线安装失败 -离线安装 github网址&#xff1a; pytorch-OpCounter:Count the MACs / FLOPs of your PyTorch model. - GitCode python setup.py install 测试&#xff1a; from options import config as c import os os.environ["CUD…

【高分系列卫星简介——高分三号卫星(GF-3)】

高分三号卫星&#xff08;GF-3&#xff09; 高分三号&#xff08;GF-3&#xff09;是我国首颗高分辨率、C频段、多极化合成孔径雷达&#xff08;SAR&#xff09;卫星&#xff0c;由中国空间技术研究院北京空间飞行器总部设计部研制&#xff0c;并于2016年8月10日成功发射。该卫…

vue实现扫雷代码复制即可用,vue2和vue3都可适用

效果预览 代码实现 <template><div id"app"><div class"mine-sweeper"><div class"board" v-for"row in board" :key"row-${row.index}"><divclass"cell":class"{ no-clickable…

Vue3:mitt实现组件通信

目录 一.性质 1.轻量级 2.单例 3.异步 4.事件绑定与解绑 二.作用 1.组件间通信 2.解耦 3.状态管理 4.事件的集中处理 三.使用 1.安装mitt 2.引入mitt&#xff1b;调用mitt&#xff1b;暴露mitt 3.组件1 4.组件2 四.代码 1.组件1 2.组件2 五.效果 一.性质 1…