[java并发]java高并发系列-第8天:线程组

原文链接:查看原文

感谢公众号“ 路人甲Java”的分享,如有冒犯,请联系删除,快去关注他吧
在这里插入图片描述

线程组

我们可以把线程归属到某个线程组中,线程组可以包含多线程线程组,线程和线程组组成了父子关系,是个树形结构,如下图:

在这里插入图片描述

使用线程组可以方便管理线程,线程组提供了一些方便我们管理线程。

创建线程关联线程组

创建线程的时候,可以给线程指定一个线程组,代码如下:

package aboutThread;

import java.util.concurrent.TimeUnit;

public class AboutThreadGroup {
    
  
        public static class R1 implements Runnable{
          
            @Override
            public void run(){
                System.out.println("当前线程名:"+Thread.currentThread().getName());
                try{
                    TimeUnit.SECONDS.sleep(1);
                }catch(InterruptedException e){
                    e.printStackTrace();
                }
            }
        }
   
    public static void main(String[] args) throws InterruptedException{
        ThreadGroup tg = new ThreadGroup("threadGroup_1");
        Thread t1 = new Thread(tg,new R1(),"t1");
        Thread t2 = new Thread(tg,new R1(),"t2");

        t1.start();
        t2.start();

        TimeUnit.SECONDS.sleep(1);

        System.out.println("活动线程数:"+tg.activeCount());
        System.out.println("活动线程组:"+tg.activeGroupCount());
        System.out.println("线程组名称:"+tg.getName());
    }
}

输出结果:

当前线程名:t2
当前线程名:t1
活动线程数:0
活动线程组:0
线程组名称:threadGroup_1

==activeCount()==方法可以返回线程组中的所有活动线程数,包含下面的所有子孙节点的线程,由于线程组中的线程是动态变化的,这个值只能是一个估算值。

为线程组指定父线程组

创建线程组的时候,可以给其指定一个父线程组,也可以不指定,如果不指定父线程组,则父线程组为当前线程的线程组,java api中有两个常用的构造方法用来创建线程组:

	public ThreadGroup(String name)
	public ThreadGroup(ThreadGroup parent,Strnig name)

第一个构造方法未指定父线程组,看一下内部实现:

public ThreadGroup()(String name){
	this(Thread.currentThread().getThreadGroup(),name);
}

系统自动获取当前线程的线程组作为默认父线程组。

代码示例:

package aboutThread;

import java.sql.Time;
import java.util.concurrent.TimeUnit;

public class AboutParentThreadGroup {
    public static class R1 implements Runnable{
        @Override
        public void run(){
            Thread thread = Thread.currentThread();
            System.out.println("所属线程组:" + thread.getThreadGroup().getName()+", 线程名称:" + thread.getName());
            try {
                TimeUnit.SECONDS.sleep(3);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }

    public static void main(String[] args) throws InterruptedException{
        ThreadGroup threadGroup1 = new ThreadGroup("thread_Group_1");
        Thread t1 = new Thread(threadGroup1,new R1(),"t1");
        Thread t2 = new Thread(threadGroup1,new R1(),"t2");
        t1.start();
        t2.start();
        TimeUnit.SECONDS.sleep(1);
        System.out.println("threadGroup1活动线程数:"+ threadGroup1.activeCount());
        System.out.println("threadGroup1活动线程组:"+ threadGroup1.activeGroupCount());
        System.out.println("threadGroup1线程组名称:"+ threadGroup1.getParent().getName());
        System.out.println("==================");
        ThreadGroup threadGroup2 = new ThreadGroup(threadGroup1,"thread_Group_2");
        Thread t3 = new Thread(threadGroup2,new R1(),"t3");
        Thread t4 = new Thread(threadGroup2,new R1(),"t4");
        t3.start();
        t4.start();
        TimeUnit.SECONDS.sleep(1);

        System.out.println("threadGroup2活动线程数" + threadGroup2.activeCount());
        System.out.println("threadGroup2活动线程组" + threadGroup2.activeGroupCount());
        System.out.println("threadGroup2线程组名称:"+ threadGroup2.getName());
        System.out.println("threadGroup2父线程组名称"+ threadGroup2.getParent().getName());

        System.out.println("=========================");

        System.out.println("threadGroup1活动线程数:"+ threadGroup1.activeCount());
        System.out.println("threadGroup1活动线程组:"+ threadGroup1.activeCount());

        System.out.println("================");

        threadGroup1.list();
    }
}

运行代码,输出:

所属线程组:thread_Group_1, 线程名称:t2
所属线程组:thread_Group_1, 线程名称:t1
threadGroup1活动线程数:2
threadGroup1活动线程组:0
threadGroup1线程组名称:main
==================
所属线程组:thread_Group_2, 线程名称:t3
所属线程组:thread_Group_2, 线程名称:t4
threadGroup2活动线程数2
threadGroup2活动线程组0
threadGroup2线程组名称:thread_Group_2
threadGroup2父线程组名称thread_Group_1
=========================
threadGroup1活动线程数:4
threadGroup1活动线程组:4
================
java.lang.ThreadGroup[name=thread_Group_1,maxpri=10]
    Thread[t1,5,thread_Group_1]
    Thread[t2,5,thread_Group_1]
    java.lang.ThreadGroup[name=thread_Group_2,maxpri=10]
        Thread[t3,5,thread_Group_2]
        Thread[t4,5,thread_Group_2]

代码解释:

  1. threadGroup1未指定父线程组,系统获取了主线程的线程组作为threadGroup1的父线程组,输出结果是:main
  2. threadGroup1为threadGroup2的父线程组
  3. threadGroup1的活动线程为4,包含了threadGoup1线程组中的t1,t2,以及子线程组threadGroup2中的t3、t4
  4. 线程组的list()方法,将线程组中的所有子孙节点信息输出到控制台,用于调试使用

根线程组

获取根线程组

	package aboutThread;

public class GetRootThreadGroup {
    
    public static void main(String[] args){
        System.out.println(Thread.currentThread());
        System.out.println(Thread.currentThread().getThreadGroup());
        System.out.println(Thread.currentThread().getThreadGroup().getParent());
        System.out.println(Thread.currentThread().getThreadGroup().getParent().getParent().getName());
    }

}

运行上面代码,输出:

Thread[main,5,main]
java.lang.ThreadGroup[name=main,maxpri=10]
java.lang.ThreadGroup[name=system,maxpri=10]
Exception in thread "main" java.lang.NullPointerException
        at aboutThread.GetRootThreadGroup.main(GetRootThreadGroup.java:9)

从上面代码可以看出:

  1. 主线程的线程组为main
  2. 根线程组为system

看一下ThreadGroup的源码:

	private ThreadGroup(){  // called from c code
		this.name = "system";
		this.maxPriority = Thread.MAX_PRIORITY;
		this.parent = null;
	}

发现ThreadGroup默认构造方法时private的,是由c调用的,创建的正是system线程组

批量停止线程

调用线程组 interrupt(),会将线程组树下的素有子孙线程中断标志置为true,可以用来批量中断线程。

示例代码:

package aboutThread;

import java.util.concurrent.TimeUnit;

public class About_ThreadGroupInterrupt {
    public static class R1 implements Runnable{
        @Override
        public void run(){
            Thread thread = Thread.currentThread();
            System.out.println("所属线程组:" + thread.getThreadGroup().getName() + ", 线程名称:"+ thread.getName());
            while(!thread.isInterrupted()){
                ;
            }
            System.out.println("线程:" + thread.getName() + "停止了!");
        }
    }

    public static void main(String[] args) throws InterruptedException{
        ThreadGroup threadGroup1 = new ThreadGroup("thread-group-1");
        Thread t1 = new Thread(threadGroup1,new R1(),"t1");
        Thread t2 = new Thread(threadGroup1,new R1(),"t2");
        t1.start();
        t2.start();

        ThreadGroup threadGroup2 = new ThreadGroup(threadGroup1,"thread-group-2");
        Thread t3 = new Thread(threadGroup2,new R1(),"t3");
        Thread t4 = new Thread(threadGroup2,new R1(),"t4");

        t3.start();
        t4.start();

        TimeUnit.SECONDS.sleep(1);

        System.out.println("----------------threadGroup1信息---------------");
        threadGroup1.list();

        System.out.println("------------------------------------------------");
        System.out.println("停止线程组:"+ threadGroup1.getName() + " 中的所有子孙线程");

        threadGroup1.interrupt();
        TimeUnit.SECONDS.sleep(2);

        System.out.println("-------------------------threadGroup1停止后,输出信息---------------------------");
        threadGroup1.list();
    }
}

运行代码,输出:

所属线程组:thread-group-2, 线程名称:t3
所属线程组:thread-group-1, 线程名称:t1
所属线程组:thread-group-2, 线程名称:t4
所属线程组:thread-group-1, 线程名称:t2
----------------threadGroup1信息---------------
java.lang.ThreadGroup[name=thread-group-1,maxpri=10]
    Thread[t1,5,thread-group-1]
    Thread[t2,5,thread-group-1]
    java.lang.ThreadGroup[name=thread-group-2,maxpri=10]
        Thread[t3,5,thread-group-2]
        Thread[t4,5,thread-group-2]
------------------------------------------------
停止线程组:thread-group-1 中的所有子孙线程
线程:t4停止了!
线程:t3停止了!
线程:t1停止了!
线程:t2停止了!
-------------------------threadGroup1停止后,输出信息---------------------------
java.lang.ThreadGroup[name=thread-group-1,maxpri=10]
    java.lang.ThreadGroup[name=thread-group-2,maxpri=10]

停止线程后,通过 list() 方法可以看出输出的信息中不包含已结束的线程了。

多说几句,建议大家在创建线程或者线程组的时候,给他们取一个有意义的名字,对于计算机来说,可能名字不重要,但实在系统出现问题的时候,你可能会查看线程堆栈信息,如果你看到的都是t1,t2,t3,估计自己也比较崩溃,如果看到的是httpAccpHandler、dubboHandler类似的名字,应该会好很多。


并发学习第8天,加油!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值