面试官:有了解过线程组和线程优先级吗?

开发 前端
通过上面的了解,我们应该知道为什么要用这种树状结构了。它都是一层一层级别的控制,这么做方便去管理,提高安全性,出了问题也能很快的定位到。就像公司的人员组织架构一样,一切都是为了管理好公司。

什么是线程组

在Java中,线程组使用ThreadGroup表示,其中Thread存于线程组中,从字面意思也很好理解。在创建线程过程中,Thread不能独立于线程组之外,之前我们学习创建线程时,没有指定线程组,因为在默认情况下,它会将当前的线程环境作为线程组, 可以通过Thread.currentThread().getThreadGroup()获取线程组,线程组可以方便管理我们的线程,一定程度上提高了安全性。

public class ThreadGroupTest {
public static void main(String[] args) {
new Thread(() -> {
System.out.println(Thread.currentThread().getThreadGroup().getName());
}).start();
System.out.println(Thread.currentThread().getThreadGroup().getName());
}
}

输出:

main
main

可以发现在main线程组下;

ThreadGroup是一个标准的「向下引用」的树状结构,这样设计的原因是「防止"上级"线程被"下级"线程引用而无法有效地被GC回收」。

线程优先级

线程的优先级级别由操作系统决定,不同的操作系统级别是不一样的,在Java中,提供了一个级别范围1~10,方便我们去参考。Java默认的线程优先级为5,线程的执行顺序由调度程序来决定,线程的优先级会在线程被调用之前设定。

源码描述:

/**
* 最低级别
*/
public final static int MIN_PRIORITY = 1;

/**
* 默认级别
*/
public final static int NORM_PRIORITY = 5;

/**
* 最高级别
*/
public final static int MAX_PRIORITY = 10;

获取线程优先级别

public static void main(String[] args) {
new Thread(() -> {
System.out.println("default level: {}" + Thread.currentThread().getPriority());
}).start();
}

输出: default level: {}5

设置级别

public static void main(String[] args) {
Thread t = new Thread(() -> {
System.out.println("default level: {}" + Thread.currentThread().getPriority());
});
t.start();
t.setPriority(10);
}

输出: default level: {}10

通常来讲,高级别的优先级往往会更高几率的执行,注意这里是概率性问题,下面我们测试一下:

public static void main(String[] args) {
Thread t = new Thread(() -> {
System.out.println("hello " + Thread.currentThread().getPriority());
System.out.println("default level: {}" + Thread.currentThread().getPriority());
});
t.setPriority(3);

Thread t1 = new Thread(() -> {
System.out.println("hello " + Thread.currentThread().getPriority());
System.out.println("default level: {}" + Thread.currentThread().getPriority());
});
t1.setPriority(7);

Thread t2 = new Thread(() -> {
System.out.println("hello " + Thread.currentThread().getPriority());
System.out.println("default level: {}" + Thread.currentThread().getPriority());
});
t2.setPriority(10);


t.start();
t1.start();
t2.start();
}

第一次输出:

hello 7
default level: {}7
hello 10
default level: {}10
hello 3
default level: {}3

第二次:

hello 3
default level: {}3
hello 7
default level: {}7
hello 10
default level: {}10

第三次

hello 10
default level: {}10
hello 7
default level: {}7
hello 3
default level: {}3

...

发现,不断的尝试之后,高级别出现的概率会比较靠前一点, 所以想要借助它来完成一些特定业务的同学注意了,不建议使用,不靠谱,之前也讲到,底层还是由操作系统调度完成

Java提供一个「线程调度器」来监视和控制处于「RUNNABLE状态」的线程。线程的调度策略采用「抢占式」,优先级高的线程比优先级低的线程会有更大的几率优先执行。在优先级相同的情况下,按照“先到先得”的原则。每个Java程序都有一个默认的主线程,就是通过JVM启动的第一个线程main线程。

除了主线程之外,还有一个线程是守护线程,它的优先级比较低。如果所有的非守护线程都结束了,这个守护线程也会自动结束。可以借助它实现一些特定场景,比如手动关闭线程的场景,某些场景下不关闭,会造成资源浪费,手动关闭又很麻烦。这里我们可以通过setDaemon(true)指定。

public static void main(String[] args) {
Thread t = new Thread(() -> {
System.out.println("hello " + Thread.currentThread().getPriority());
System.out.println("default level: {}" + Thread.currentThread().getPriority());
});
t.setDaemon(true); // 默认为false
t.setPriority(10);

Thread t1 = new Thread(() -> {
System.out.println("hello " + Thread.currentThread().getPriority());
System.out.println("default level: {}" + Thread.currentThread().getPriority());
});
t1.setPriority(7);

t.start();
t1.start();
}

输出:

hello 7
default level: {}7
hello 10
default level: {}10

发现即使指定了高级别,执行的优先级仍然是最低的

线程组下的优先级

刚刚我们都是在main线程组下,举一反三,线程组下的优先级又是怎么样的呢❓下面,测试一下:

public static void main(String[] args) {
// 指定 name 为 g1的线程组
ThreadGroup group = new ThreadGroup("g1");
group.setMaxPriority(4);

Thread t = new Thread(group, () -> {
System.out.println("hello " + Thread.currentThread().getPriority());
System.out.println("default level: {}" + Thread.currentThread().getPriority());
}, "t0");
t.setPriority(10);
t.start();
}

输出:

hello 4
default level: {}4

发现,在g1线程组下指定了最大优先级后,线程t0的优先级最大级别只能是4, 所以这也是使用线程组的好处。

我们可以通过如下方式复制线程组, ThreadGroup提供了enumerate方法:

public static void main(String[] args) throws InterruptedException {
// 指定 name 为 g1的线程组
ThreadGroup group = new ThreadGroup("g1");
group.setMaxPriority(4);

Thread t = new Thread(group, () -> {
System.out.println("hello " + Thread.currentThread().getPriority());
System.out.println("default level: {}" + Thread.currentThread().getPriority());
}, "t0");
t.setPriority(10);
t.start();

// 复制线程组
System.out.println(group.activeCount()); // 1
Thread[] list = new Thread[group.activeCount()];
group.enumerate(list);

Thread.sleep(3000);
System.out.println(list[0].getName()); // 输出 t0

}

统一异常捕获

public static void main(String[] args) {
// 指定 name 为 g1的线程组
ThreadGroup group = new ThreadGroup("g1") {
// 统一异常捕获
public void uncaughtException(Thread t, Throwable e) {
System.out.println(t.getName() + ": " + e.getMessage()); // t0: 我出错了
}
};
group.setMaxPriority(4);

Thread t = new Thread(group, () -> {
System.out.println("hello " + Thread.currentThread().getPriority());
System.out.println("default level: {}" + Thread.currentThread().getPriority());
throw new RuntimeException("我出错了");
}, "t0");
t.setPriority(10);
t.start();
}

向下引用的树状数据结构

线程组的内部其实不单单可以放线程,其实也可以放其它线程组,我们看下源码定义

 public static void main(String[] args) {
// 指定 name 为 g1的线程组
ThreadGroup group = new ThreadGroup("g1") {
// 统一异常捕获
public void uncaughtException(Thread t, Throwable e) {
System.out.println(t.getName() + ": " + e.getMessage()); // t0: 我出错了
}
};
group.setMaxPriority(4);

Thread t = new Thread(group, () -> {
System.out.println("hello " + Thread.currentThread().getPriority());
System.out.println("default level: {}" + Thread.currentThread().getPriority());
throw new RuntimeException("我出错了");
}, "t0");
t.setPriority(10);
t.start();
}

这里大家可以大胆去猜测一下,为什么要采用这种数据结构❓其实你通过源码发现,它的内部很多地方都调用了checkAccess方法,特别是在set操作,字面意思是检查是否有权限,我看下这个方法。

public final void checkAccess() {
SecurityManager security = System.getSecurityManager();
if (security != null) {
security.checkAccess(this);
}
}

它调用了一个SecurityManager, 它是Java的安全管理器,它允许应用程序在执行一个可能不安全或敏感的操作前确定该操作是什么,以及是否是在允许执行该操作的安全上下文中执行它。应用程序可以允许或不允许该操作。总的来说就是保证安全性。

通过上面的了解,我们应该知道为什么要用这种树状结构了。它都是一层一层级别的控制,这么做方便去管理,提高安全性,出了问题也能很快的定位到。就像公司的人员组织架构一样,一切都是为了管理好公司。

结束语

本篇内容到这里就结束了, 大家自己一定要多去理解,不要去背,。

责任编辑:武晓燕 来源: 今日头条
相关推荐

2022-06-24 06:43:57

线程池线程复用

2022-08-02 06:31:32

Java并发工具类

2022-07-26 08:40:42

Java并发工具类

2009-08-28 17:10:59

C#线程优先级

2024-04-10 09:47:59

Java调度虚拟线程

2022-07-11 10:47:46

容器JAVA

2024-05-20 10:03:15

线程池优先级队列排序方法

2010-03-18 14:09:20

Java线程同步

2024-09-11 22:51:19

线程通讯Object

2022-06-10 13:56:42

Java

2022-02-08 08:14:07

Context数据线程

2024-04-02 09:45:27

线程池Executors开发

2023-10-12 07:35:45

面试线程通信

2020-06-04 08:36:55

Linux内核线程

2023-12-20 14:35:37

Java虚拟线程

2024-03-11 18:18:58

项目Spring线程池

2024-09-09 15:09:30

2022-06-30 08:14:05

Java阻塞队列

2023-06-30 19:44:56

CPU调频线程

2021-06-11 07:26:17

NodeJSCommonJSRequire
点赞
收藏

51CTO技术栈公众号