优雅的停止线程
- 在多线程操作之中如果要启动多线程,肯定要使用Thread类中的start()方法,而如果对于对线程进行停止处理Thread类原本提供stop()方法,但是对于这个方法从JDK1.2就已经将其废除了,而且一直到现在,也不建议再用此方法,除了stop()之外还有几个方法被禁用了:
- 停止多线程:public void stop()
- 销毁多线程:public void destroy()
- 挂起线程:public final void suspend()、暂停执行
- 恢复挂起的线程执行:public final void resume()
- 之所以废除这些方法,主要原因是这些方法有可能会导致线程的死锁,所以从JDK1.2开始就不建议使用了,这个时候要想实现线程的停止需要一种柔和的方式来进行;
//实现线程柔和的停止
package com.company;
public class ThreadStop {
public static boolean flag = true;
public static void main(String[] args) throws Exception {
new Thread(() -> {
long num = 0;
while (flag) {
try {
Thread.sleep(50);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName() + "正在执行,num = " + num++);
}
}, "线程").start();
Thread.sleep(200);
flag = false;
}
}
- 如果有其他线程去控制这个flag的内容,那么这个时候对于线程的停止也不是立刻就停止的,而是会在执行中判断flag的内容来完成;
后台守护线程
- 多线程里面可以进行守护线程的定义,如果主线程的程序或者其他线程的程序还在执行的时候,那么守护线程将一直存在,并且运行在后台状态;
- 在Thread类里面提供有如下守护线程的操作方法:
- 设置为守护线程:public final void setDaemon(boolean on)
- 判断是否为守护线程:public final boolean isDaemon()
//使用守护线程
package com.company;
public class GuardianThread {
public static void main(String[] args) {
Thread userThread = new Thread(() -> {
for (int x = 0; x < 10; x ++) {
System.out.println(Thread.currentThread().getName() + "、x = " + x);
}
}, "用户线程");
Thread daemonThread = new Thread(() -> {
for (int x = 0; x < Integer.MAX_VALUE; x ++) {
System.out.println(Thread.currentThread().getName() + "、x = " + x);
}
}, "守护线程");
daemonThread.setDaemon(true);
userThread.start();
daemonThread.start();
}
}
- 守护线程都是围绕在用户线程的周围,如果守护线程执行完毕了,守护线程也就消失了,在JVM里面最大的守护线程就是GC线程;
- 程序执行中GC线程会一直存在,如果程序执行完毕,GC线程也将消失;
volatile关键字
- 在多线程定义之中,volatile关键字主要是在属性定义上使用的,表示此属性为直接数据操作,而不进行副本的拷贝处理;