iis服务器助手广告广告
返回顶部
首页 > 资讯 > 后端开发 > Python >java高并发之线程组详解
  • 497
分享到

java高并发之线程组详解

2024-04-02 19:04:59 497人浏览 独家记忆

Python 官方文档:入门教程 => 点击学习

摘要

目录线程组创建线程关联线程组为线程组指定父线程组根线程组批量停止线程总结线程组 我们可以把线程归属到某个线程组中,线程组可以包含多个线程以及线程组,线程和线程组组成了父子关系,是个树

线程组

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

640?wx_fmt=png

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

创建线程关联线程组

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


package com.itsoku.chat02;	
import java.util.concurrent.TimeUnit;	
	
public class Demo1 {	
    public static class R1 implements Runnable {	
        @Override	
        public void run() {	
            System.out.println("threadName:" + Thread.currentThread().getName());	
            try {	
                TimeUnit.SECONDS.sleep(3);	
            } catch (InterruptedException e) {	
                e.printStackTrace();	
            }	
        }	
    }	
    public static void main(String[] args) throws InterruptedException {	
        ThreadGroup threadGroup = new ThreadGroup("thread-group-1");	
        Thread t1 = new Thread(threadGroup, new R1(), "t1");	
        Thread t2 = new Thread(threadGroup, new R1(), "t2");	
        t1.start();	
        t2.start();	
        TimeUnit.SECONDS.sleep(1);	
        System.out.println("活动线程数:" + threadGroup.activeCount());	
        System.out.println("活动线程组:" + threadGroup.activeGroupCount());	
        System.out.println("线程组名称:" + threadGroup.getName());	
    }	
}

输出结果:

threadName:t1
threadName:t2
活动线程数:2
活动线程组:0
线程组名称:thread-group-1

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

为线程组指定父线程组

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


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

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


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

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

上一段示例代码:


package com.itsoku.chat02;	
import java.util.concurrent.TimeUnit;	
	
public class Demo2 {	
    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.getName());	
        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.activeGroupCount());	
        System.out.println("----------------------");	
        threadGroup1.list();	
    }	
}

输出结果:

所属线程组:thread-group-1,线程名称:t1
所属线程组:thread-group-1,线程名称:t2
threadGroup1活动线程数:2
threadGroup1活动线程组:0
threadGroup1线程组名称:thread-group-1
threadGroup1父线程组名称:main
----------------------
所属线程组:thread-group-2,线程名称:t4
所属线程组:thread-group-2,线程名称:t3
threadGroup2活动线程数:2
threadGroup2活动线程组:0
threadGroup2线程组名称:thread-group-2
threadGroup2父线程组名称:thread-group-1
----------------------
threadGroup1活动线程数:4
threadGroup1活动线程组:1
----------------------
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,包含了threadGroup1线程组中的t1、t2,以及子线程组threadGroup2中的t3、t4

4.线程组的list()方法,将线程组中的所有子孙节点信息输出到控制台,用于调试使用

根线程组

获取根线程组


package com.itsoku.chat02;	
	
public class Demo3 {	
    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());	
    }	
}

运行上面代码,输出:

Thread[main,5,main]
java.lang.ThreadGroup[name=main,maxpri=10]
java.lang.ThreadGroup[name=system,maxpri=10]
null

从上面代码可以看出:

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 com.itsoku.chat02;	
import java.util.concurrent.TimeUnit;	
	
public class Demo4 {	
    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-1,线程名称:t1
所属线程组:thread-group-1,线程名称:t2
所属线程组:thread-group-2,线程名称:t3
所属线程组:thread-group-2,线程名称:t4
-----------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停止了!
线程:t2停止了!
线程:t1停止了!
线程:t3停止了!
----------threadGroup1停止后,输出信息------------
java.lang.ThreadGroup[name=thread-group-1,maxpri=10]
java.lang.ThreadGroup[name=thread-group-2,maxpri=10]

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

总结

本篇文章就到这里了,希望能够给你带来帮助,也希望您能够多多关注编程网的更多内容!

--结束END--

本文标题: java高并发之线程组详解

本文链接: https://www.lsjlt.com/news/155554.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • java高并发之线程组详解
    目录线程组创建线程关联线程组为线程组指定父线程组根线程组批量停止线程总结线程组 我们可以把线程归属到某个线程组中,线程组可以包含多个线程以及线程组,线程和线程组组成了父子关系,是个树...
    99+
    2022-11-12
  • java高并发之线程的基本操作详解
    目录新建线程终止线程线程中断等待(wait)和通知(notify)挂起(suspend)和继续执行(resume)线程等待线程结束(join)和谦让(yeild)总结新建线程 新建线...
    99+
    2022-11-12
  • java的多线程高并发详解
    目录1.JMM数据原子操作2.来看volatile关键字3.并发编程三大特性4.双锁判断机制创建单例模式5.synchronized关键字6.AtomicIntger原子操作7.锁优...
    99+
    2022-11-12
  • 详解Java高并发编程之AtomicReference
    目录一、AtomicReference 基本使用1.1、使用 synchronized 保证线程安全性二、了解 AtomicReference2.1、使用 AtomicReferen...
    99+
    2022-11-12
  • java高并发之理解进程和线程
    目录进程线程进程与线程的一个简单解释总结进程 进程(Process)是计算机中的程序关于某数据集合上的一次运行活动,是系统进行资源分配和调度的基本单位,是操作系统结构的基础。程序是指...
    99+
    2022-11-12
  • Java并发编程之详解CyclicBarrier线程同步
    CyclicBarrier线程同步 java.util.concurrent.CyclicBarrier提供了一种多线程彼此等待的同步机制,可以把它理解成一个障碍,所有先到达这个障碍...
    99+
    2022-11-12
  • java 多线程与并发之volatile详解分析
    目录CPU、内存、缓存的关系CPU缓存什么是CPU缓存为什么要有多级CPU CacheJava内存模型(Java Memory Model,JMM)JMM导致的并发安全问题可见性原子...
    99+
    2022-11-12
  • java高并发的用户线程和守护线程详解
    目录程序只有守护线程时,系统会自动退出设置守护线程,需要在start()方法之前进行线程daemon的默认值总结守护线程是一种特殊的线程,在后台默默地完成一些系统性的服务,比如垃圾回...
    99+
    2022-11-12
  • Java多线程之并发编程的核心AQS详解
    目录一、AQS简介1.1、AOS概念 1.2、AQS的核心思想1.3、AQS是自旋锁1.4、AQS支持两种资源分享的方式 二、AQS原理2.1、同步状态的管理2....
    99+
    2022-11-12
  • 详解Java多线程与并发
    目录一、进程与线程二、并发与并行1、线程安全问题2、共享内存不可见性问题三、创建线程1、继承Thread类2、实现Runable接口3、实现Callable接口四、Thread类详解...
    99+
    2022-11-12
  • Java并发编程之ThreadLocal详解
    目录一、什么是ThreadLocal?二、ThreadLocal的使用场景三、如何使用ThreadLocal四、数据库连接时的使用五、ThreadLocal工作原理六、小结七、注意点...
    99+
    2022-11-12
  • 理解Java多线程之并发编程
    目录1 多线程的使用场景2 多线程的缺点2.1 上下文切换的开销(1)上下文切换的开销(2)如何减少上下文切换2.2 多线程中的数据一致性问题(1)线程中访问外部数据的过程(2)线程...
    99+
    2023-02-02
    Java并发编程 java并发编程实战 java并发编程的艺术
  • 【Java|多线程与高并发】定时器(Timer)详解
    文章目录 1. 前言2. 定时器的基本使用3. 实现定时器4. 优化上述的定时器代码5. 总结 1. 前言 在Java中,定时器Timer类是用于执行定时任务的工具类。它允许你安排一个...
    99+
    2023-10-05
    java jvm 开发语言
  • Java高并发之CyclicBarrier的用法详解
    目录使用方式注意事项总结Java 中的 CyclicBarrier 是一种同步工具,它可以让多个线程在一个屏障处等待,直到所有线程都到达该屏障处后,才能继续执行。CyclicBarr...
    99+
    2023-03-13
    Java高并发CyclicBarrier Java高并发 Java CyclicBarrier
  • Java面试必备之JMM高并发编程详解
    目录一、什么是JMM二、JMM定义了什么原子性可见性有序性三、八种内存交互操作四、volatile关键字可见性volatile一定能保证线程安全吗禁止指令重排序volatile禁止指...
    99+
    2022-11-13
  • Java并发编程之Executors类详解
    一、Executors的理解 Executors类属于java.util.concurrent包; 线程池的创建分为两种方式:ThreadPoolExecutor ...
    99+
    2022-11-12
  • Java并发编程之LockSupport类详解
    目录一、LockSupport类的属性二、LockSupport类的构造函数三、park(Object blocker)方法 和 park()方法分析四、parkNanos(Obje...
    99+
    2022-11-12
  • Java并发编程之详解ConcurrentHashMap类
    前言 由于Java程序员常用的HashMap的操作方法不是同步的,所以在多线程环境下会导致存取操作数据不一致的问题,Map接口的另一个实现类Hashtable 虽然是线程安全的,但是...
    99+
    2022-11-12
  • Java多线程之并发编程的基石CAS机制详解
    目录一、CAS机制简介1.1、悲观锁和乐观锁更新数据方式1.2、什么是CAS机制1.3、CAS与sychronized比较1.4、Java中都有哪些地方应用到了CAS机制呢?...
    99+
    2022-11-12
  • Java多线程并发之ReentrantLock
    目录ReentrantLock公平锁和非公平锁重入锁小结疑惑ReentrantLock 公平锁和非公平锁 这个类是接口 Lock的实现类,也是悲观锁的一种,但是它提供了 lock和 ...
    99+
    2023-05-18
    Java 多线程并发 Java ReentrantLock
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作