Thread daemonThread = new Thread(() -> IntStream.rangeClosed(1, 100000)
.forEach(System.out::println));
daemonThread.setDaemon(true);
daemonThread.start();
Thread.sleep(10);
System.out.println("End of the execution in the Thread " +
Thread.currentThread().getName());
}
}
在这个例子中,我使用了守护程序线程来声明1到100,000的范围,迭代所有这些,然后打印。但请记住,如果非守护进程的主线程首先完成,守护程序线程将无法完成执行。
输出将按如下方式进行:
- 在主线程中开始执行。
- 打印数字从1到100,000。
- 主线程中的执行结束,很可能在迭代到100,000之前完成。
最终输出将取决于你的JVM实现。
事实证明:线程是不可预测的。
线程优先级和JVM
可以使用该setPriority方法确定线程执行的优先级,但是如何处理它取决于JVM实现。Linux,MacOS和Windows都有不同的JVM实现,每个都将根据自己的默认值处理线程优先级。
但是,你设置的线程优先级确实会影响线程调用的顺序。在Thread类上的三个常数是:
/**
- The minimum priority that a thread can have.
*/
public static final int MIN_PRIORITY = 1;
/**
- The default priority that is assigned to a thread.
*/
public static final int NORM_PRIORITY = 5;
/**
- The maximum priority that a thread can have.
*/
public static final int MAX_PRIORITY = 10;
尝试对以下代码运行一些测试,以查看最终的执行优先级
public class ThreadPriority {
public static void main(String… threadPriority) {
Thread moeThread = new Thread(() -> System.out.println(“Moe”));
Thread barneyThread = new Thread(() -> System.out.println(“Barney”));
Thread homerThread = new Thread(() -> System.out.println(“Homer”));
moeThread.setPriority(Thread.MAX_PRIORITY);
barneyThread.setPriority(Thread.NORM_PRIORITY);
homerThread.setPriority(Thread.MIN_PRIORITY);
homerThread.start();
barneyThread.start();
moeThread.start();
}
}
即使我们设置moeThread为MAX_PRIORITY,我们也不能指望首先执行此线程。相反,执行顺序将是随机的。
但是,使用常量有一个问题:如果传递的优先级数字不在1到10的范围内,setPriority()方法将引发IllegalArgumentException。所以我们可以使用枚举来解决这个问题。使用枚举Enums既简化了代码,又能够更好地控制代码的执行。
Java线程挑战!
我们已经了解了不少关于线程的知识,如果你想更进一步,研究以下代码:
public class ThreadChallenge {
private static int line = 10;
public static void main(String… doYourBest) {
new tool(“Car”).start();
tool Bike = new tool(“Bike”);
Bike.setPriority(Thread.MAX_PRIORITY);
Bike.setDaemon(false);
Bike.start();
tool Train = new tool(“Train”);
Train.setPriority(Thread.MIN_PRIORITY);
Train.start();
}
static class tool extends Thread {
tool(String Name) { super(Name); }
@Override public void run() {
line++;
if (line == 13) {
System.out.println(this.getName());
}
}
}
}
这段代码的输出是什么?根据你上面学的内容分析代码。
A. Car
B. Bike
C. Train
D. 不确定
解析上述代码,了解多线程的运行
在上面的代码中,我们创建了三个线程。第一个线程是Car,我们为此线程分配了默认优先级。第二个线程是Bike,分配了分配了MAX_PRIORITY优先级。第二个线程是Train,分配了MIN_PRIORITY优先级。然后我们开始了多线程。为了确定线程将运行的顺序,您可能首先注意到tool类扩展了Thread类,并且我们已经在构造函数中传递了线程名称。我们还run()中如果line等于13就进行打印。注意!即使Train是我们执行顺序中的第三个线程,并且MIN_PRIORITY不能保证它将在所有JVM实现的最后执行。您可能还会注意到,在此示例中,我们将Bike线程设置为守护,所以它是一个守护程序线程,Bike可能永远不会完成执行。但是其他两个线程默认是非守护进程,因此Car和Train线程肯定会完成它们的执行。总之,结果将是D:不确定,因为无法保证线程调度程序将遵循我们的执行顺序或线程优先级。请记住,如果不借助JUC的工具,我们不能依赖程序逻辑(线程或线程优先级的顺序)来预测线程的执行顺序。
多线程常见错误
- 调用run()方法以尝试启动新线程。
- 试图启动一个线程两次(这将导致一个IllegalThreadStateException)。
- 允许多个进程在不应更改时更改对象的状态。
- 编写依赖于线程优先级的程序逻辑
- 依赖于线程执行的顺序 - 即使我们首先启动一个线程,也不能保证它将首先被执行。
总结
最后对于程序员来说,要学习的知识内容、技术有太多太多,要想不被环境淘汰就只有不断提升自己,从来都是我们去适应环境,而不是环境来适应我们!
这里附上上述的技术体系图相关的几十套腾讯、头条、阿里、美团等公司2021年的面试题,把技术点整理成了视频和PDF(实际上比预期多花了不少精力),包含知识脉络 + 诸多细节,由于篇幅有限,这里以图片的形式给大家展示一部分。
相信它会给大家带来很多收获:
上述【高清技术脑图】以及【配套的架构技术PDF】可以关注我免费获取
oid%E5%BC%80%E5%8F%91%E4%B8%8D%E4%BC%9A%E8%BF%99%E4%BA%9B%EF%BC%9F%E5%A6%82%E4%BD%95%E9%9D%A2%E8%AF%95%E6%8B%BF%E9%AB%98%E8%96%AA%EF%BC%81.md)
当程序员容易,当一个优秀的程序员是需要不断学习的,从初级程序员到高级程序员,从初级架构师到资深架构师,或者走向管理,从技术经理到技术总监,每个阶段都需要掌握不同的能力。早早确定自己的职业方向,才能在工作和能力提升中甩开同龄人。