线程组,顾名思义,就是线程的组,逻辑类似项目组,用于管理项目成员,线程组就是用来管理线程。
每个线程都会有一个线程组,如果没有设置将会有些默认的初始化设置
而在java中线程组则是使用类ThreadGroup 进行抽象描述
既然线程组是用来管理线程的,自然更多的是一种管理维度的抽象,所以很多方法也都是这个理念

构造方法

想要了解一个类的具体信息,第一个要看的就是构造方法,看一下最多的内容的那个构造方法就可以大致了解到有哪些属性了
ThreadGroup有两个构造方法
仔细看下这两个构造方法,其实只有一个了,那就是底层的私有的这一个
对于一个线程组来说,他拥有他自己的名字,也拥有他的优先级,也有是否是守护的说法
不同于线程,对于线程组来说,他是有记录自己的父线程组的,通过parent
另外,线程组也记录了自己下面有哪些线程组,使用数组记录,也就是构造方法中的  parent.add(this)
所以一个线程组核心的信息是:名称、优先级、是否守护、父线程组、子线程组
另外还有一个默认的构造方法,看注释,用来创建系统线程组
  

名称

线程组的名称借助于内部的name属性持有
通过构造方法可以设置名称
提供了get方法用于获取名称

优先级

此处的优先级,表示的是最大允许优先级,线程组内最大就允许这么大
里面所有的线程不能继续变大,不要认为是记录了里面所有的线程中最大的那个值,是一个天花板,不是一个记录尺

daemon

父线程组

对于线程组来说,是明确的记录了他的父
借助于parent这个属性值,可以获取一个线程组的父线程组,也可以用来确定是否是一个指定线程组的父或者祖先

子线程组

内部借助于ThreadGroup 数组维护内部的线程组,从这个数据组织结构来看,就很显然,线程组内可以有线程组,可以层层嵌套形成树状结构的
对于线程组的创建,他必然会有一个父线程组(不设置就是当前线程所在的线程组了,也可以简单说当前线程组)
创建线程组的时候,就会借助于add方法,将这个线程组加入到父线程组维护的数组内
对于任何一个线程,也都是拥有一个线程组,如果没有设置,将会将当前线程的线程组作为线程组,这个在前面已经说过
而在start方法中,又将当前线程添加到了线程组,请看下面的源码截图
在回头看下这个add方法,借助于内部的线程数组,其实就是将这个线程添加到数组内
  • nThreads 记录的就是线程组内部的线程个数
  • nUnstartedThreads记录的是未启动的个数
刚刚调用线程的start方法,这个数就要减1,尽管可能这个瞬间线程可能并没有真正的启动,确保能够明确的声明线程组内有启动的线程了
所以就由这几项数据组成了线程组的树形结构
也就是说
  • 每个线程组也都知道自己包含多少个线程,哪些线程;
  • 每个线程组也都知道自己包含了多少个线程组,哪些线程组;
这是一份很重要的信息,借助于这些信息就完全串联起来了

子线程组相关方法

既然是树形结构,那么自然可能有枚举节点的需求
ThreadGroup中提供了两类enumerate方法,看名字应该就可以理解含义了,用于枚举线程和线程组

线程枚举

底层依赖于私有枚举方法,把此线程组及其子组中的所有活动线程复制到指定数组中。可以设置是否递归枚举
两个方法中,如果不指定是否递归,那么默认是递归的,他们都将参数数组的第一个元素开始写入(0号下标)
很显然,他们内部就是借助于树结构的变量,nThreads和thread[]数组
需要注意的是,如果数组内空间不足,多余的线程将不能够保存进去,而且保存的是alive状态的
 

activeCount

该线程组以及子线程组中,活动线程的估计数。注意是一个估计数,估计数,估计数

activeGroupCount

类似activeCount,这个方法是返回的线程组的个数,仍旧是估计数,估计数,估计数

list方法

list看注释,用于调试,底层依赖方法list(PrintStream out, int indent),indent表示的是缩进,也就是空格个数
仍旧是借助于nthreads和ngroups以及threads数组和group数组,也就是树形结构循环遍历打印信息

interrupt()方法

中断此线程组中的所有线程,可以看得出来:
仍旧是遍历树形结构,核心是调用所有线程的interrupt方法
所以,此方法是中断该线程组以及所有子线程组中的所有线程

线程组的销毁

线程组的销毁内部借助于boolean变量 destroyed 进行标识
getter方法,isDestroyed直接返回此字段
而setter方法destroy,也是设置这个字段,但是还有一些逻辑判断与处理
destroy()负责销毁此线程组及其所有子组。
  • 会进行权限校验
  • 并且此线程组必须为空,也就是nthreads > 0不成立,也就是此线程组中的所有线程都已停止执行。
  • 并且会将子线程组中的也进行销毁,是递归进行的,显然,如果子线程组中线程非空,那么仍旧会抛出异常

权限校验checkAccess

checkAccess就是总提到的一个借助于安全管理器进行权限校验的封装
确定当前运行的线程是否有权修改此线程组

异常处理器

uncaughtException,是用于异常处理的设置,此处不讲,后续单独章节。

总结

从前面的描述可以看得出来,线程组就是对线程进行管理的一个抽象构建,他包括了自身的一些信息,还有一大部分就是对于线程的管理
线程组中有线程,也有线程组,借助于两个变量和两个数组完成了树形结构的构造,很多方法都是借助于这个树形结构完成的,比如枚举
想要理解线程组,就要理解线程组“管理”角色的内涵,并且对线程组的树形结构了解
既然是管理线程,所以线程中的一些功能或者属性也是依赖线程组的,比如优先级,线程不能超过线程组的最大优先级,再比如Thread中的activeCount(),实际上就是currentThread().getThreadGroup().activeCount();
总之,一定要理解管理二字的含义

线程组ThreadGroup分析详解 多线程中篇(三)的更多相关文章

  1. Java线程创建形式 Thread构造详解 多线程中篇(五)

    Thread作为线程的抽象,Thread的实例用于描述线程,对线程的操纵,就是对Thread实例对象的管理与控制. 创建一个线程这个问题,也就转换为如何构造一个正确的Thread对象. 构造方法列表 ...

  2. InheritableThreadLocal类原理简介使用 父子线程传递数据详解 多线程中篇(十八)

      上一篇文章中对ThreadLocal进行了详尽的介绍,另外还有一个类: InheritableThreadLocal 他是ThreadLocal的子类,那么这个类又有什么作用呢?   测试代码 p ...

  3. Thread类详解 多线程中篇(二)

    Java.lang.Thread是Java应用程序员对Java多线程的第一站,Thread就是对Java线程本身的抽象 所以在Java中的线程编程概念中,一个Thread实例 == 一个线程 线程有哪 ...

  4. “全栈2019”Java多线程第十三章:线程组ThreadGroup详解

    难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java多 ...

  5. “全栈2019”Java多线程第二十二章:饥饿线程(Starvation)详解

    难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java多 ...

  6. “全栈2019”Java多线程第九章:判断线程是否存活isAlive()详解

    难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java多 ...

  7. “全栈2019”Java多线程第七章:等待线程死亡join()方法详解

    难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java多 ...

  8. “全栈2019”Java多线程第五章:线程睡眠sleep()方法详解

    难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java多 ...

  9. 线程池底层原理详解与源码分析(补充部分---ScheduledThreadPoolExecutor类分析)

    [1]前言 本篇幅是对 线程池底层原理详解与源码分析  的补充,默认你已经看完了上一篇对ThreadPoolExecutor类有了足够的了解. [2]ScheduledThreadPoolExecut ...

随机推荐

  1. 如何使用$.each()与$().each()以及他们的区别

    1.首先,说下$.each(Arry/Object,function(index,val){ //index表示下标,val表示下标对应的值 }) 下面是使用$.each()的几种类型,其中arr2与 ...

  2. 非vue-cli的花括号闪现问题

    <div id="app" v-cloak></div>[v-cloak] { display: none;}这种方式可以解决网速较慢,vue.js文件还没 ...

  3. HTML入门12

    开始了解响应式图片 响应式,根据屏幕尺寸和分辨率的设备上都能良好工作以及其他特性的图片,接下来考虑怎样创建自适应得图片,专注于img元素,完成自适应. 分辨率切换,不同的尺寸 <img srcs ...

  4. hadoop环境搭建及Wordcount案例实验

    1.Linux环境变量设置 在/etc/profile中添加环境变量 sudo vim /etc/profile PATH=$PATH:/usr/local/hadoop/bin source /et ...

  5. 初始化git库并配置自动部署

    1.初始化库 git init --bare wap.git 2.配置wap.git/config文件 [core] repositoryformatversion = 0 filemode = tr ...

  6. js根据顺序加载,有依赖关系

    function loadScript(url, callback) { var script = document.createElement("script"); script ...

  7. NeuChar 平台使用及开发教程(一):开始使用 NeuChar

    如果您还不知道什么是 NeuChar,或者我们为什么要做 NeuChar,请看这里. 角色 NeuChar 打通了从开发者到终端用户的完整的上下游链路,了解 NeuChar 之前,先要明确两个角色身份 ...

  8. SQL 查询当前时间

    Mysql: select date_format(now(),'%Y-%m-%d'); Oracle: Oracle中如何获取系统当前时间进行语句的筛选是SQL语句的常见功能 获取系统当前时间dat ...

  9. 255.Spring Boot+Spring Security:使用md5加密

    说明 (1)JDK版本:1.8 (2)Spring Boot 2.0.6 (3)Spring Security 5.0.9 (4)Spring Data JPA 2.0.11.RELEASE (5)h ...

  10. 消息队列RabbitMq、ActiveMq、ZeroMq、kafka之间的比较

    MQ框架非常之多,比较流行的有RabbitMq.ActiveMq.ZeroMq.kafka.这几种MQ到底应该选择哪个?要根据自己项目的业务场景和需求.下面我列出这些MQ之间的对比数据和资料. 第一部 ...