最后
看完上述知识点如果你深感Java基础不够扎实,或者刷题刷的不够、知识不全面
小编专门为你量身定制了一套<Java一线大厂高岗面试题解析合集:JAVA基础-中级-高级面试+SSM框架+分布式+性能调优+微服务+并发编程+网络+设计模式+数据结构与算法>
针对知识面不够,也莫慌!还有一整套的<Java核心进阶手册>,可以瞬间查漏补缺
全都是一丢一丢的收集整理纯手打出来的
更有纯手绘的各大知识体系大纲,可供梳理:Java筑基、MySQL、Redis、并发编程、Spring、分布式高性能架构知识、微服务架构知识、开源框架知识点等等的xmind手绘图~
当前程序中,除了main主线程外没有其他非守护线程的线程了,因此,main线程结束之后,所有守护线程也将结束。
还有一个值得去注意的点:如果所有非守护线程的线程结束,守护线程也将结束,守护线程中finally块中的代码也不会执行(这个可以自行检验一下),因此不能依赖守护线程完成清理或者收尾工作,因为你完全不知道自己下一秒守护线程是否还健在。
需要注意
-
由守护线程创建的线程默认也是守护线程,由用户线程创建的也就是用户线程。
-
在线程启动(start)之后,不允许将线程设置为守护线程(setDaemon),否则将会抛出
java.lang.IllegalThreadStateException
异常。
优先级
我们说过,各个处于就绪状态线程等待资源调度是按照一定规则的,这个规则就是线程拥有的优先级。
以下参考《Java编程思想》:
JDK有10个优先级,但它和大多数操作系统都不能很好地进行映射。例如Windows有7个优先级且不是固定的,所以这种映射关系也是不确定的。所以在调整优先级的时候,使用下面三种常量,具有更好的移植性。
//线程可有的最小优先级
public final static int MIN_PRIORITY = 1;
//默认优先级
public final static int NORM_PRIORITY = 5;
//线程可有的最大优先级
public final static int MAX_PRIORITY = 10;
调度器会倾向于让优先级较高的线程先执行,但并不意味着优先级较低的线程将得不到执行。
进行试验:
- 定义一个实现Runnable接口的类。
class PDemo implements Runnable {
@Override
public void run() {
String name = Thread.currentThread().getName();
for (int i = 0; i < 10; i++) {
System.out.println(name + “:” + i);
}
}
}
- 创建两个线程对象。
Thread t1 = new Thread(new PDemo(),“A”);
Thread t2 = new Thread(new PDemo(),“B”);
- 可以通过线程对象的
getPriority()
方法获取当前优先级。
//默认情况下,线程的优先级为5
System.out.println(t1.getPriority());//5
System.out.println(t2.getPriority());//5
- 通过线程对象的
setPriority(int newPriority)
方法设置优先级。
//设置线程优先级
t1.setPriority(Thread.MIN_PRIORITY);
t2.setPriority(Thread.MAX_PRIORITY);
- 调用线程对象的start()方法启动线程。
t1.start();
t2.start();
-
每个线程默认的优先级都与创建它的父线程的优先级的相同。
-
main线程的优先级是
NORM_PRIORITY
为5。
弃用三兄弟
stop()
resume
suspend
中断三兄弟
interrupt()
public void interrupt()
是Thread类的一个实例方法,说是说用来中断线程,但其实只是给线程设置了一个"中断"标志(true) ,线程仍然会继续运行,用户可以监视线程的状态并做出相应处理。
官方文档是这么说的:
线程调用interrupt()
将会把标志位设置为true,除此之外,情况不同,处理不同:
- 如果这个线程由于wait(),join(),sleep()等方法陷入等待状态时,它的中断状态将被清除(也就是true重新变为false),而且会收到一个
InterruptedException
。
但是我按照下面代码测试了一下,join()和sleep()都能成功检验,但是wait()检验不出,不知问题出在哪,评论区大神求助!!
下面这俩目前还没有接触到,以后有机会做总结:
-
如果这个线程由于
java.nio.channels.InterruptibleChannel
中的IO操作发生阻塞,线程还将收到一个ClosedByInterruptException
。 -
如果这个线程在
Selector
中被阻塞,它可能带有一个非零值,从选择操作立即返回,就像调用了选择器的wakeup()
方法一样。
interrupted()
public static boolean interrupted()
是静态方法,内部调用当前线程的isInterrupted方
法,会重置当前线程的中断状态。也就是说,如果线程被设置为中断标志,第一次调用此方法将会返回true,并将中断标志重置,第二次调用该方法,将会返回false。
isInterrupted()
public boolean isInterrupted()
是实例方法,测试当前线程的对象是否被中断,而不会重置当前线程的中断状态。
安全终止线程
以下内容参考:《Java并发编程的艺术》
上面提到,interrupt()
方法只是给线程标志为“中断”状态,并不会让线程真正中断,我们可以对标识位进行监测并做出相应处理,比如,我们可以通过中断操作与自定义变量来控制是否需要停止任务并终止该线程。
定义一个线程内部类。
private static class Runner implements Runnable {
private long i;
//定义变量作为标识位,用volatile修饰,自身拥有可见性和原子性
private volatile boolean on = true;
@Override
public void run() {
//对自定义标识位以及中断标识进行校验
while (on && !Thread.currentThread().isInterrupted()) {
i++;
}
System.out.println("Count i = " + i);
}
//取消操作
public void cancel() {
on = false;
}
}
利用标识位优雅地中断或结束线程。
public static void main(String[] args) throws InterruptedException {
Runner one = new Runner();
Thread countThread = new Thread(one, “CountThread”);
countThread.start();
//睡眠一秒,main线程对CountThread进行中断,使CountThread能够感知中断而结束
TimeUnit.SECONDS.sleep(1);
countThread.interrupt();
Runner two = new Runner();
最后
分享一些资料给大家,我觉得这些都是很有用的东西,大家也可以跟着来学习,查漏补缺。
《Java高级面试》
《Java高级架构知识》
《算法知识》
Java高级面试》**
[外链图片转存中…(img-yZe7chnu-1715642163501)]
《Java高级架构知识》
[外链图片转存中…(img-EsVnUZlw-1715642163501)]
《算法知识》
[外链图片转存中…(img-bzYNLn81-1715642163502)]