您的位置:首页 > 编程语言 > Java开发

Java多线程(三)——线程挂起、恢复、终止

2016-03-01 15:57 489 查看
挂起和恢复线程

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 多线程