Qu'il s'agisse d'un thread ou d'un pool de threads dans le langage Java , la valeur par défaut est le fil utilisateur , les fils utilisateur sont donc également appelés fils de discussion ordinaires.
En prenant le thread comme exemple, si vous souhaitez vérifier si un thread est un thread démon, il vous suffit d'interroger en appelant la méthode isDaemon()
si la requête. la valeur est false signifie qu'il ne s'agit pas d'un thread démon et qu'il appartient naturellement au thread utilisateur <code>isDaemon()
方法查询即可,如果查询的值为 false
则表示不为守护线程,自然也就属于用户线程了,
如下代码所示:
public static void main(String[] args) throws InterruptedException { Thread thread = new Thread(new Runnable() { @Override public void run() { System.out.println("我是子线程"); } }); System.out.println("子线程==守护线程:" + thread.isDaemon()); System.out.println("主线程==守护线程:" + Thread.currentThread().isDaemon()); }
以上程序的执行结果为:
从上述结果可以看出,默认情况下主线程和创建的新线程都为用户线程。
PS:Thread.currentThread() 的意思是获取执行当前代码的线程实例。
守护线程(Daemon Thread)也被称之为后台线程或服务线程,守护线程是为用户线程服务的,当程序中的用户线程全部执行结束之后,守护线程也会跟随结束。
守护线程的角色就像“服务员”,而用户线程的角色就像“顾客”,当“顾客”全部走了之后(全部执行结束),那“服务员”(守护线程)也就没有了存在的意义,所以当一个程序中的全部用户线程都结束执行之后,那么无论守护线程是否还在工作都会随着用户线程一块结束,整个程序也会随之结束运行。
那如何将默认的用户线程修改为守护线程呢?
这个问题要分为两种情况来回答,首先如果是线程,则可以通过设置 setDaemon(true)
方法将用户线程直接修改为守护线程,而如果是线程池则需要通过 ThreadFactory
将线程池中的每个线程都为守护线程才行,接下来我们分别来实现一下。
如果使用的是线程,可以通过 setDaemon(true)
方法将线程类型更改为守护线程,如下代码所示:
public static void main(String[] args) throws InterruptedException { Thread thread = new Thread(new Runnable() { @Override public void run() { System.out.println("我是子线程"); } }); // 设置子线程为守护线程 thread.setDaemon(true); System.out.println("子线程==守护线程:" + thread.isDaemon()); System.out.println("主线程==守护线程:" + Thread.currentThread().isDaemon()); }
以上程序的执行结果为:
要把线程池设置为守护线程相对来说麻烦一些,需要将线程池中的所有线程都设置成守护线程,这个时候就需要使用 ThreadFactory
来定义线程池中每个线程的线程类型了,具体实现代码如下:
// 创建固定个数的线程池 ExecutorService threadPool = Executors.newFixedThreadPool(10, new ThreadFactory() { @Override public Thread newThread(Runnable r) { Thread t = new Thread(r); // 设置线程为守护线程 t.setDaemon(false); return t; } });
如下图所示:
如上图所示,可以看出,整个程序中有 10 个守护线程都是我创建的。其他几种创建线程池的设置方式类似,都是通过 ThreadFactory
统一设置的,这里就不一一列举了。
通过前面的学习我们可以创建两种不同的线程类型了,那二者有什么差异呢?接下来我们使用一个小示例来看一下。
下面我们创建一个线程,分别将这个线程设置为用户线程和守护线程,在每个线程中执行一个 for
循环,总共执行 10 次信息打印,每次打印之后休眠 100 毫秒,来观察程序的运行结果。
新建的线程默认就是用户线程,因此我们无需对线程进行任何特殊的处理,执行 for
循环即可(总共执行 10 次信息打印,每次打印之后休眠 100 毫秒),实现代码如下:
/** * Author:Java中文社群 */ public class DaemonExample { public static void main(String[] args) throws InterruptedException { Thread thread = new Thread(new Runnable() { @Override public void run() { for (int i = 1; i <= 10; i++) { // 打印 i 信息 System.out.println("i:" + i); try { // 休眠 100 毫秒 Thread.sleep(100); } catch (InterruptedException e) { e.printStackTrace(); } } } }); // 启动线程 thread.start(); } }
以上程序执行结果如下:
从上述结果可以看出,当程序执行完 10 次打印之后才会正常结束进程。
/** * Author:Java中文社群 */ public class DaemonExample { public static void main(String[] args) throws InterruptedException { Thread thread = new Thread(new Runnable() { @Override public void run() { for (int i = 1; i <= 10; i++) { // 打印 i 信息 System.out.println("i:" + i); try { // 休眠 100 毫秒 Thread.sleep(100); } catch (InterruptedException e) { e.printStackTrace(); } } } }); // 设置为守护线程 thread.setDaemon(true); // 启动线程 thread.start(); } }
以上程序执行结果如下:
从上述结果可以看出,当线程设置为守护线程之后,整个程序不会等守护线程 for
public static void main(String[] args) throws InterruptedException { Thread thread = new Thread(new Runnable() { @Override public void run() { for (int i = 1; i <= 10; i++) { // 打印 i 信息 System.out.println("i:" + i + ",isDaemon:" + Thread.currentThread().isDaemon()); try { // 休眠 100 毫秒 Thread.sleep(100); } catch (InterruptedException e) { e.printStackTrace(); } } } }); // 启动线程 thread.start(); // 设置为守护线程 thread.setDaemon(true); }
#🎜🎜#PS : Thread.currentThread() signifie obtenir l'instance de thread qui exécute le code actuel. #🎜🎜##🎜🎜#2. Changez-le activement en fil de démon #🎜🎜##🎜🎜#Daemon Thread (Daemon Thread) est également appelé fil d'arrière-plan ou fil de service, #🎜🎜#Daemon. thread Il sert les threads utilisateur Lorsque tous les threads utilisateur du programme sont exécutés, le thread démon se terminera également. #🎜🎜##🎜🎜#Le rôle du thread démon est comme un "serveur", et le rôle du thread utilisateur est comme un "client". Lorsque tous les "clients" partent (toute exécution se termine), le " waiter" (thread démon) ) n'a aucun sens d'existence, donc lorsque tous les threads utilisateur d'un programme terminent leur exécution, peu importe si le thread démon fonctionne toujours ou non, il se terminera avec les threads utilisateur, et le programme entier sera finissent également de courir. #🎜🎜##🎜🎜##🎜🎜#Alors comment changer le thread utilisateur par défaut en thread démon ? #🎜🎜##🎜🎜##🎜🎜#Cette question peut recevoir une réponse dans deux situations. Premièrement, s'il s'agit d'un fil de discussion, vous pouvez directement modifier le fil de l'utilisateur en définissant le
setDaemon(true)
. Méthode thread démon, et s'il s'agit d'un pool de threads, vous devez utiliser ThreadFactory
pour faire de chaque thread du pool de threads un thread démon. Ensuite, nous l'implémenterons séparément. #🎜🎜#setDaemon(true)< /code> method,# 🎜🎜#Le code suivant est affiché : #🎜🎜##🎜🎜#<div class="code" style="position:relative; padding:0px; margin:0px;"><div class="code" style="position:relative; padding:0px; margin:0px;"><pre class="brush:java;">public static void main(String[] args) throws InterruptedException {
Thread thread = new Thread(new Runnable() {
@Override
public void run() {
Thread thread2 = new Thread(new Runnable() {
@Override
public void run() {
}
});
System.out.println("守护线程的子线程 thread2 isDaemon:" +
thread2.isDaemon());
}
});
// 设置为守护线程
thread.setDaemon(true);
// 启动线程
thread.start();
Thread.sleep(1000);
}</pre><div class="contentsignin">Copier après la connexion</div></div><div class="contentsignin">Copier après la connexion</div></div>#🎜🎜##🎜🎜#Le résultat de l'exécution du programme ci-dessus est : #🎜🎜##🎜🎜## 🎜🎜#<img src="https ://img.php.cn/upload/article/000/887/227/168394848910534.png" alt="Comment utiliser les threads utilisateur et les threads démon en Java" />#🎜 🎜#<h4>2.2 Définir le pool de threads sur le thread démon</h4>#🎜🎜#Il est relativement difficile de définir le pool de threads en tant que thread démon. Tous les threads du pool de threads doivent être définis comme des threads démon. dans ce cas, vous devez utiliser <code>ThreadFactory</code > Pour définir le type de thread de chaque thread dans le pool de threads, le code d'implémentation spécifique est le suivant : #🎜🎜#<div class="code" style="position:relative; padding:0px; margin:0px;"><div class="code" style="position:relative; padding:0px; margin:0px;"><pre class="brush:java;">public static void main(String[] args) throws InterruptedException {
Thread thread = new Thread(new Runnable() {
@Override
public void run() {
for (int i = 1; i <= 10; i++) {
// 打印 i 信息
System.out.println("i:" + i);
try {
// 休眠 100 毫秒
Thread.sleep(100);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
});
// 设置为守护线程
thread.setDaemon(true);
// 启动线程
thread.start();
// 等待线程执行完
thread.join();
System.out.println("子线程==守护线程:" + thread.isDaemon());
System.out.println("主线程==守护线程:" + Thread.currentThread().isDaemon());
}</pre><div class="contentsignin">Copier après la connexion</div></div><div class="contentsignin">Copier après la connexion</div></div>#🎜🎜##🎜🎜# as ci-dessous : #🎜🎜##🎜🎜##🎜🎜#< img src="https://img.php.cn/upload/article/000/887/227/168394848954238.png" alt="Comment utiliser threads utilisateur et threads démons en Java" />#🎜🎜##🎜🎜 #Comme le montre l'image ci-dessus, on peut voir qu'il y a 10 threads démons dans l'ensemble du programme créé par moi. Plusieurs autres paramètres de création de pools de threads sont similaires et ils sont tous définis uniformément via <code>ThreadFactory
, je ne les listerai donc pas un par un ici. #🎜🎜##🎜🎜#3. Thread démon VS thread utilisateur #🎜🎜##🎜🎜#Grâce à l'apprentissage précédent, nous pouvons créer deux types de threads différents. Jetons ensuite un coup d'œil à l'aide d'un petit exemple. #🎜🎜##🎜🎜# Ensuite, nous créons un thread, définissons ce thread comme thread utilisateur et thread démon respectivement, exécutons une boucle for
dans chaque thread et exécutons l'impression des informations 10 fois au total. pendant 100 millisecondes après chaque impression pour observer les résultats d'exécution du programme. #🎜🎜#pour
boucle (Un total de 10 impressions d'informations sont effectuées, en veille pendant 100 millisecondes après chaque impression), #🎜🎜#Le code d'implémentation est le suivant : #🎜🎜##🎜🎜#public class DaemonExample { private static final int count = 100000; public static void main(String[] args) throws InterruptedException { // 定义任务 Runnable runnable = new Runnable() { @Override public void run() { for (int i = 0; i < count; i++) { System.out.println("执行线程:" + Thread.currentThread().getName()); } } }; // 创建守护线程 t1 Thread t1 = new Thread(runnable, "t1"); // 设置为守护线程 t1.setDaemon(true); // 启动线程 t1.start(); // 创建用户线程 t2 Thread t2 = new Thread(runnable, "t2"); // 启动线程 t2.start(); } }
public class DaemonExample { private static final int count = 100000; public static void main(String[] args) throws InterruptedException { // 定义任务 Runnable runnable = new Runnable() { @Override public void run() { for (int i = 0; i < count; i++) { System.out.println("执行线程:" + Thread.currentThread().getName()); } } }; // 创建守护线程 t1 Thread t1 = new Thread(runnable, "t1"); // 设置为守护线程 t1.setDaemon(true); // 启动线程 t1.start(); // 创建用户线程 t2 Thread t2 = new Thread(runnable, "t2"); // 设置 t2 的优先级为最高 t2.setPriority(Thread.MAX_PRIORITY); // 启动线程 t2.start(); } }
for
boucle 10 fois avant de se fermer. Au lieu de cela, lorsque le thread principal se termine. , le thread démon n'exécute la boucle qu'une seule fois, puis se termine. À partir de là, nous pouvons voir la différence entre les threads démon et les threads utilisateur. #🎜🎜#守护线程是为用户线程服务的,当一个程序中的所有用户线程都执行完成之后程序就会结束运行,程序结束运行时不会管守护线程是否正在运行,由此我们可以看出守护线程在 Java 体系中权重是比较低的。
守护线程的使用需要注意以下三个问题:
守护线程的设置 setDaemon(true)
必须要放在线程的 start()
之前,否则程序会报错。
在守护线程中创建的所有子线程都是守护线程。
使用 jojn()
方法会等待一个线程执行完,无论此线程是用户线程还是守护线程。
接下来我们分别演示一下,以上的注意事项。
当我们将 setDaemon(true)
设置在 start()
之后,如下代码所示:
public static void main(String[] args) throws InterruptedException { Thread thread = new Thread(new Runnable() { @Override public void run() { for (int i = 1; i <= 10; i++) { // 打印 i 信息 System.out.println("i:" + i + ",isDaemon:" + Thread.currentThread().isDaemon()); try { // 休眠 100 毫秒 Thread.sleep(100); } catch (InterruptedException e) { e.printStackTrace(); } } } }); // 启动线程 thread.start(); // 设置为守护线程 thread.setDaemon(true); }
以上程序执行结果如下:
从上述结果可以看出,当我们将 setDaemon(true)
设置在 start()
之后,不但程序的执行会报错,而且设置的守护线程也不会生效。
public static void main(String[] args) throws InterruptedException { Thread thread = new Thread(new Runnable() { @Override public void run() { Thread thread2 = new Thread(new Runnable() { @Override public void run() { } }); System.out.println("守护线程的子线程 thread2 isDaemon:" + thread2.isDaemon()); } }); // 设置为守护线程 thread.setDaemon(true); // 启动线程 thread.start(); Thread.sleep(1000); }
以上程序执行结果如下:
从上述结果可以看出,守护线程中创建的子线程,默认情况下也属于守护线程。
通过 3.2 部分的内容我们可以看出,默认情况下程序结束并不会等待守护线程执行完,而当我们调用线程的等待方法 join()
时,执行的结果就会和 3.2 的结果有所不同,下面我们一起来看吧,
示例代码如下:
public static void main(String[] args) throws InterruptedException { Thread thread = new Thread(new Runnable() { @Override public void run() { for (int i = 1; i <= 10; i++) { // 打印 i 信息 System.out.println("i:" + i); try { // 休眠 100 毫秒 Thread.sleep(100); } catch (InterruptedException e) { e.printStackTrace(); } } } }); // 设置为守护线程 thread.setDaemon(true); // 启动线程 thread.start(); // 等待线程执行完 thread.join(); System.out.println("子线程==守护线程:" + thread.isDaemon()); System.out.println("主线程==守护线程:" + Thread.currentThread().isDaemon()); }
以上程序执行结果如下:
通过上述结果我们可以看出,即使是守护线程,当程序中调用 join()
方法时,程序依然会等待守护线程执行完成之后再结束进程。
守护线程的典型应用场景就是垃圾回收线程,当然还有一些场景也非常适合使用守护线程,比如服务器端的健康检测功能,对于一个服务器来说健康检测功能属于非核心非主流的服务业务,像这种为了主要业务服务的业务功能就非常合适使用守护线程,当程序中的主要业务都执行完成之后,服务业务也会跟随者一起销毁。
首先来说,线程的类型(用户线程或守护线程)并不影响线程执行的优先级,如下代码所示,定义一个用户线程和守护线程,分别执行 10 万次循环,通过观察最后的打印结果来确认线程类型对程序执行优先级的影响。
public class DaemonExample { private static final int count = 100000; public static void main(String[] args) throws InterruptedException { // 定义任务 Runnable runnable = new Runnable() { @Override public void run() { for (int i = 0; i < count; i++) { System.out.println("执行线程:" + Thread.currentThread().getName()); } } }; // 创建守护线程 t1 Thread t1 = new Thread(runnable, "t1"); // 设置为守护线程 t1.setDaemon(true); // 启动线程 t1.start(); // 创建用户线程 t2 Thread t2 = new Thread(runnable, "t2"); // 启动线程 t2.start(); } }
以上程序执行结果如下:
通过上述结果可以看出,线程的类型不管是守护线程还是用户线程对程序执行的优先级是没有任何影响的,而当我们将 t2
的优先级调整为最大时,整个程序的运行结果就完全不同了,
如下代码所示:
public class DaemonExample { private static final int count = 100000; public static void main(String[] args) throws InterruptedException { // 定义任务 Runnable runnable = new Runnable() { @Override public void run() { for (int i = 0; i < count; i++) { System.out.println("执行线程:" + Thread.currentThread().getName()); } } }; // 创建守护线程 t1 Thread t1 = new Thread(runnable, "t1"); // 设置为守护线程 t1.setDaemon(true); // 启动线程 t1.start(); // 创建用户线程 t2 Thread t2 = new Thread(runnable, "t2"); // 设置 t2 的优先级为最高 t2.setPriority(Thread.MAX_PRIORITY); // 启动线程 t2.start(); } }
以上程序执行结果如下:
00000000 通过上述的结果可以看出,程序的类型和程序执行的优先级是没有任何关系,当新创建的线程默认的优先级都是 5 时,无论是守护线程还是用户线程,它们执行的优先级都是相同的,当将二者的优先级设置不同时,执行的结果也会随之改变(优先级设置的越高,最早被执行的概率也越大)。
Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!