多线程
上篇是线程的生命周期,那这次来分享下,终止线程的几种方式。
1、正常运行结束
所谓正常运行结束,我的理解就是程序正常运行结束,线程自动结束。
2. 使用退出标志退出线程
一般run()方法执行完,线程就会正常结束,然而腻,常常有些线程是伺服线程。他们需要长时间的运行,只有在外部某些条件满足的情况下,才能关闭这些线程。使用一个变量来控制循环,例如:最直接的方法就是设一个boolean类型的标志,并通过设置这个标志位true或false来控制while循环是否退出,下面放代码:
public class ThreadSafe extends Thread {
public volatile boolean exit = false;
public void run() {
while (!exit){
}
}
}
这里定义了一个退出标志exit,当exit为true时,while循环退出,exit的默认值为false。在定义exit时使用了一个Java关键字volatile,这个关键字的目的是使exit同步,也就是说在同一时刻只能由一个线程来修改exit的值。
3、Interrupt方法结束线程
使用interrupt()方法来中断线程是分两种情况的:
- 线程处于阻塞状态:如使用了sleep,同步锁的wait(),socket中的receiver(),accept()等方法时,会使线程处于阻塞状态。当调用线程interrupt()方法时,会抛出InterruptException异常。阻塞中的那个方法抛出这个异常,通过代码可以捕获该异常,然后跳出循环状态,从而让我们有机会结束这个线程的执行。通常有些小伙伴认为只要调用interrupt方法,线程就会结束,实际上是不正确的,一定要先捕获InterruptException异常之后通过break来跳出循环,才能正常结束run()方法。
- 线程未处于阻塞状态:使用isInterrupted()判断线程的中断标志来退出循环。当使用interrupt()方法时,中断标志就会为true,和使用自定义的标志来控制循环是一样的道理。
public class ThreadSafe extends Thread {
public void run() {
while(!isInterrupted()) {
try {
Thread.sleep(5*1000);
} catch(InterruptedException e) {
e.printStackTrace();
break;
}
}
}
}
4.stop()方法终止线程(不安全)
这个各位小伙伴应该在学习中都有用过吧,我们可以直接使用thread.stop()来强行终止线程,但是stop()方法是很危险的,就想突然关闭电源一样。这样子可能机会产生不可预料的结果,不安全主要是:
- thread.stop()调用之后,创建子线程的线程就会抛出ThreadDeatherror的错误,并且会释放子线程所持有的所有锁。一般任何进行加锁的代码块,都是为了保护数据的一致性,如果在调用thread.stop()后导致了该线程所持有的所有的锁突然释放,那么被保护的数据就可能呈现不一致性,其他线程在使用这些被破坏的数据时,有可能导致一些很奇怪的应用程序错误。因此,并不推荐使用stop()方法来终止线程。
下面补充一些常用方法的区别:
1.sleep()和wait()区别:
- 对于sleep()方法,我们首先要知道该方法是属于Thread类中的,而wait()方法,则是属于Object类中的。
- sleep()方法导致了程序暂停执行指定的时间,让出CPU给其他线程,但是它依然处于监控状态,当指定时间到了又会自动恢复运行状态。
- 调用sleep()方法的过程中,线程不会释放对象锁。
- 调用wait()方法的时候,线程会释放对象锁,进入等待此对象的等待锁定池,只有针对此对象调用notify()方法后本线程才进入对象锁定池准备获取对象锁进入运行状态。
2.start()和run()的区别
- start()方法来启动线程,真正实现了多线程运行。这时无需等待run()方法体代码执行完毕,可以继续执行下面的代码。
- 通过调用Thread类的start()方法来启动一个线程,此时线程是处于就绪状态,并没有运行。
- 方法run()称为线程体,它包含了要执行的这个线程的内容,次线程终止。然后CPU在调度其他线程。
Java后台线程
- 定义:守护线程–也称“服务线程”,他是后台线程,它有一个特性,即为用户线程 提供 公共服务,在没有用户线程可服务时会自动离开。
- 优先级:守护线程的优先级比较低,用于为系统中的其它对象和线程提供服务。
- 设置:通过setDaemon(true)来设置线程为“守护线程”;将一个用户线程设置为守护线程的方式是在 线程对象创建 之前 用线程对象的setDaemon方法。
- 在Daemon线程中产生的新线程也是Daemon的。
- 线程则是JVM级别的,以Tomcat 为例,如果你在Web 应用中启动一个线程,这个线程的生命周期并不会和Web应用程序保持同步。也就是说,即使你停止了Web应用,这个线程依旧是活跃的。
- example: 垃圾回收线程就是一个经典的守护线程,当我们的程序中不再有任何运行的Thread,程序就不会再产生垃圾,垃圾回收器也就无事可做,所以当垃圾回收线程是JVM上仅剩的线程时,垃圾回收线程会自动离开。它始终在低级别的状态中运行,用于实时监控和管理系统中的可回收资源。
- 生命周期:守护进程(Daemon)是运行在后台的一种特殊进程。它独立于控制终端并且周期性地执行某种任务或等待处理某些发生的事件。也就是说守护线程不依赖于终端,但是依赖于系统,与系统“同生共死”。当JVM中所有的线程都是守护线程的时候,JVM就可以退出了;如果还有一个或以上的非守护线程则JVM不会退出。
总结
希望对你有所帮助哦,如果有什么错误还请大佬指出,再此感谢。
|