Java多线程(三)——线程挂起、恢复、终止
2016-03-01 15:57
489 查看
挂起和恢复线程
suspend()方法容易发生死锁。调用suspend()的时候,目标线程会停下来,但却仍然持有在这之前获得的锁定。此 时,其他任何线程都不能访问锁定的资源,除非被”挂起”的线程恢复运行。对任何线程来说,如果它们想恢复目标线程,同时又试图使用任何一个锁定的资源,就 会造成死锁。
所以不应该使用suspend(),而应在自己的Thread类中置入一个标志,指出线程应该活动还是挂起。若标志指出线程应该挂起,便用wait()命其进入等待状态。若标志指出线程应当恢复,则用一个notify()重新启动线程。
正确方式:
可以看出程序运行良好
错误方式:
执行结果如下:
终止线程
反对使用stop(),是因为它不安全。它会解除由线程获取的所有锁定,而且如果对象处于一种不连贯状态,那么其他线程能在那种状态下检查和修改它们。
终止线程的替代方法:同样是使用标志位,通过控制标志位来终止线程。
suspend()方法容易发生死锁。调用suspend()的时候,目标线程会停下来,但却仍然持有在这之前获得的锁定。此 时,其他任何线程都不能访问锁定的资源,除非被”挂起”的线程恢复运行。对任何线程来说,如果它们想恢复目标线程,同时又试图使用任何一个锁定的资源,就 会造成死锁。
所以不应该使用suspend(),而应在自己的Thread类中置入一个标志,指出线程应该活动还是挂起。若标志指出线程应该挂起,便用wait()命其进入等待状态。若标志指出线程应当恢复,则用一个notify()重新启动线程。
正确方式:
public class AlternateSuspendResume extends Object implements Runnable { private volatile int firstVal; private volatile int secondVal; //增加标志位,用来实现线程的挂起和恢复 private volatile boolean suspended; public boolean areValuesEqual() { return ( firstVal == secondVal ); } public void run() { try { suspended = false; firstVal = 0; secondVal = 0; workMethod(); } catch ( InterruptedException x ) { System.out.println("interrupted while in workMethod()"); } } private void workMethod() throws InterruptedException { int val = 1; while ( true ) { //仅当贤臣挂起时,才运行这行代码 waitWhileSuspended(); stepOne(val); stepTwo(val); val++; //仅当线程挂起时,才运行这行代码 waitWhileSuspended(); Thread.sleep(200); } } private void stepOne(int newVal) throws InterruptedException { firstVal = newVal; Thread.sleep(300); } private void stepTwo(int newVal) { secondVal = newVal; } public void suspendRequest() { suspended = true; } public void resumeRequest() { suspended = false; } private void waitWhileSuspended() throws InterruptedException { //这是一个“繁忙等待”技术的示例。 //它是非等待条件改变的最佳途径,因为它会不断请求处理器周期地执行检查, //更佳的技术是:使用Java的内置“通知-等待”机制 while ( suspended ) { Thread.sleep(200); } } public static void main(String[] args) { AlternateSuspendResume asr = new AlternateSuspendResume(); Thread t = new Thread(asr); t.start(); //休眠1秒,让其他线程有机会获得执行 try { Thread.sleep(1000); } catch ( InterruptedException x ) { } for ( int i = 0; i < 10; i++ ) { asr.suspendRequest(); //让线程有机会注意到挂起请求 //注意:这里休眠时间一定要大于 //stepOne操作对firstVal赋值后的休眠时间,即300ms, //目的是为了防止在执行asr.areValuesEqual()进行比较时, //恰逢stepOne操作执行完,而stepTwo操作还没执行 try { Thread.sleep(350); } catch ( InterruptedException x ) { } System.out.println("dsr.areValuesEqual()=" + asr.areValuesEqual()); asr.resumeRequest(); try { //线程随机休眠0~2秒 Thread.sleep( ( long ) (Math.random() * 2000.0) ); } catch ( InterruptedException x ) { //略 } } System.exit(0); //退出应用程序 } }
可以看出程序运行良好
错误方式:
public class MyThreadDemo extends Object implements Runnable { // volatile关键字,表示该变量可能在被一个线程使用的同时,被另一个线程修改 private volatile int firstVal; private volatile int secondVal; // 判断二者是否相等 public boolean areValuesEqual() { return (firstVal == secondVal); } @Override public void run() { try { firstVal = 0; secondVal = 0; workMethod(); } catch (InterruptedException x) { System.out.println("interrupted while in workMethod()"); } } private void workMethod() throws InterruptedException { int val = 1; while (true) { stepOne(val); stepTwo(val); val++; Thread.sleep(200); // 再次循环钱休眠200毫秒 } } // 赋值后,休眠300毫秒,从而使线程有机会在stepOne操作和stepTwo操作之间被挂起 private void stepOne(int newVal) throws InterruptedException { firstVal = newVal; Thread.sleep(300); // 模拟长时间运行的情况 } private void stepTwo(int newVal) { secondVal = newVal; } public static void main(String[] args) { MyThreadDemo dsr = new MyThreadDemo(); Thread t = new Thread(dsr); t.start(); // 休眠1秒,让其他线程有机会获得执行 try { Thread.sleep(1000); } catch (InterruptedException x) { } for (int i = 0; i < 10; i++) { // 挂起线程 t.suspend(); System.out.println("dsr.areValuesEqual()=" + dsr.areValuesEqual()); // 恢复线程 t.resume(); try { // 线程随机休眠0~2秒 Thread.sleep((long) (Math.random() * 2000.0)); } catch (InterruptedException x) { // 略 } } System.exit(0); // 中断应用程序 } }
执行结果如下:
dsr.areValuesEqual()=true dsr.areValuesEqual()=true dsr.areValuesEqual()=true dsr.areValuesEqual()=false dsr.areValuesEqual()=false dsr.areValuesEqual()=false dsr.areValuesEqual()=false dsr.areValuesEqual()=true dsr.areValuesEqual()=true dsr.areValuesEqual()=false
终止线程
反对使用stop(),是因为它不安全。它会解除由线程获取的所有锁定,而且如果对象处于一种不连贯状态,那么其他线程能在那种状态下检查和修改它们。
终止线程的替代方法:同样是使用标志位,通过控制标志位来终止线程。
相关文章推荐
- java对世界各个时区(TimeZone)的通用转换处理方法(转载)
- java-注解annotation
- java-模拟tomcat服务器
- java-用HttpURLConnection发送Http请求.
- java-WEB中的监听器Lisener
- Android IPC进程间通讯机制
- Android Native 绘图方法
- Android java 与 javascript互访(相互调用)的方法例子
- Python3写爬虫(四)多线程实现数据爬取
- 介绍一款信息管理系统的开源框架---jeecg
- 聚类算法之kmeans算法java版本
- java实现 PageRank算法
- PropertyChangeListener简单理解
- c++11 + SDL2 + ffmpeg +OpenAL + java = Android播放器
- 插入排序
- 冒泡排序
- 堆排序
- 快速排序