【JavaEE初阶 — 多线程】Thread的常见构造方法&属性


目录

Thread类的属性

[1.Thread 的常见构造方法](#1.Thread 的常见构造方法)

[2.Thread 的几个常见属性](#2.Thread 的几个常见属性)

[2.1 前台线程与后台线程](#2.1 前台线程与后台线程)

[2.2 setDaemon()](#2.2 setDaemon())

[2.3 isAlive()](#2.3 isAlive())


Thread类的属性


  • Thread 类是JVM 用来管理线程的一个类,换句话说,每个线程都有一个唯一的Thread 对象与之关联;
  • 每个执行流,也需要有一个对象来描述;
  • Thread类的对象就是用来描述一个线程执行流的,JVM会将这些Thread对象组织起来,用于线程调度,线程管理。

1.Thread 的常见构造方法


|--------------------------------------------|-------------------------------------------------------|
| 方法 | 说明 |
| Thread() | 创建线程对象,必须要重写 Thread类 的 run()方法 |
| Thread(Runnable target) | 使用 Runnable 对象创建线程对象 【不用重写 Thread类 的 run()方法】 |
| Thread(String name) | 创建线程对象,并命名 |
| Thread(Runnable target,String name) | 使用 Runnable对象创建线程对象,并命名 |
| Thread(ThreadGroup group, Runnable target) | 线程可以被用来分组管理,分好的组即为线程组 【这个目前我们了解即可】 |


对于 Thread类 的第四种构造方法,第一个参数传的是lamda表达式,lamda相当于Rannable;

再传入的第二个参数,就是这个线程的名字,需要手动添加**" "**

如果要调用:带有 自定义线程名字 的参数 的 构造方法,无论自定义的名字是什么,都不会不影响线程的执行的;为了方便程序员调试,可以给线程起名字。

程序运行后,我们可以通过 jconsole 这个工具,在线程列表中,看到自定义名字,并且正在运行的线程。除了自定义的线程,线程列表中的线程是JVM提供的线程。


如果我们不定义名字,程序调用 Thread类 的第一个构造方法,在jconsole的线程列表中,我们可以发现,如果不自定义线程名,系统会使用默认线程名 Thread - 数字 来命名线程。


如果在一个进程 中有多个线程,这些线程,会被CPU随机调度,并发执行;

所以不一定会因为主线程结束,整个进程也跟着结束。


2.Thread 的几个常见属性


|----------------------------------------------|---------------------|
| 属性 | 获取方法 |
| ID 【线程的唯一标识,不同线程不会重复】 | getId() |
| 名称 【各种调试工具都会用到】 | getName() |
| 状态 【表示线程当前所处的一个情况】 | getState() |
| 优先级 【优先级高的线程理论上来说更容易被调度到】 | getPriority() |
| 是否后台线程 【JVM会在一个进程的所有非后台线程结束后,才会结束运行】 | isDaemon() |
| 是否存活 【简单的理解为,run() 是否运行结束了】 | isAlive() |
| 是否被中断 | isInterrupted() |


2.1 前台线程与后台线程


如果在程序运行的过程中,我们通过 jconsole 查看线程列表中没有main,说明主线程已经结束;

主线程结束,但是 t1,t2,t3 还在,使用进程还在;说明 t1,t2,t3 能够影响进程是否终止。

所有类似 t1,t2,t3 这样能够有效进程存在的线程,称为 "前台线程"。

程序员通过代码创建的线程 和 主线程,都默认是前台线程。

总结:

  1. 能影响进程存在的,是前台线程;不能影响进程存在的,是后台线程;
  2. JVM会在一个进程的所有非后台线程结束后,才会结束运行

2.2 setDaemon()


程序员通过编写代码,创建的线程,包括main主线程,默认都是前台线程;

但是可以通过setDaemon(),将 前台线程 修改成 后台线程~~

上述代码的逻辑:主线程和 t 线程并发执行,并且主线程在打印三次" hello main "后,打印结束日志,而 t 线程继续执行死循环,所以进程没有因为主线程终止而结束。

所以,t 线程是一个前台线程,我们可以通过 isDeamon() 将 t 设置成后台线程;

通过 t.isDaemon(true) ,把 t 线程 从前台线程设置成后台线程;

此时,代码中只有主线程一个前台线程,所以在主线程运行结束后,整个进程结束,t 线程的死循环也被迫结束。

拓展:

进程与进程之间存在父子关系,而线程与线程之间则不存在;比如:

IDEA 本身也是一个 Java 进程;在 IDEA 中运行一个 Java 代码,通过IDEA 进程,又创建出一个新的Java 进程,这俩进程之间就是父子关系。


2.3 isAlive()


Java 代码中创建的Thread对象,和系统中的线程,是一 一对应的关系;

但是,Thread 对象 的生命周期,和 系统中的线程 的生命周期,是不同的~~

(可能存在:Thread对象 还存活,但是 系统中的线程 已经销毁 的情况)

我们引入一段代码,来看看"Thread对象还存活,但是系统中的线程已经销毁的情况"这种情况:

这个代码的逻辑:

前三秒 sleep1000毫秒 t线程的状态都是alive的状态,三秒后执行结束,t线程的isalive状态为false

线程的入口方法里的逻辑结束了,系统中对应的线程也就随之销毁了(操作系统) ;

虽然系统中的线程已经销毁,但是 Thread对象 还存活;

因此,t线程 被销毁,但是 t 这个对象依旧存在,所以才能一直通过t对象,调用isAlive方法


再多执行几次上述代码,发现程序运行结果中的 true 的个数是 3 还是 4 不固定,这是线程随机调度造成的:

主线程第四次打印,和 t线程 的结束,谁先谁后,是不一定的;

如果打印 true 的次数为4,说明在 第四次 主线程 和 t线程 并发执行的顺序是:

  1. 先执行主线程中的打印 t.isAlive
  2. 然后才执行 t线程 中的循环判断是否结束

所以 打印的第一个 false 是第五次执行线程,此时 t 线程已经凉透了~~


相关推荐
不爱学习的YY酱16 分钟前
【操作系统不挂科】<CPU调度(13)>选择题(带答案与解析)
java·linux·前端·算法·操作系统
zongzi_49421 分钟前
二次封装的天气时间日历选择组件
开发语言·javascript·ecmascript
丁总学Java21 分钟前
Maven项目打包,com.sun.tools.javac.processing
java·maven
kikyo哎哟喂31 分钟前
Java 代理模式详解
java·开发语言·代理模式
duration~37 分钟前
SpringAOP模拟实现
java·开发语言
小码ssim44 分钟前
IDEA使用tips(LTS✍)
java·ide·intellij-idea
一条晒干的咸魚1 小时前
【Web前端】实现基于 Promise 的 API:alarm API
开发语言·前端·javascript·api·promise
就爱六点起1 小时前
C/C++ 中的类型转换方式
c语言·开发语言·c++
我明天再来学Web渗透1 小时前
【SQL50】day 2
开发语言·数据结构·leetcode·面试
猫猫的小茶馆1 小时前
【C语言】指针常量和常量指针
linux·c语言·开发语言·嵌入式软件