使用 suspend 和 resume 暂停和恢复线程

suspend 和 resume 的使用

在 Thread 类中有这样两个方法:suspend 和 resume,这两个方法是成对出现的。

  • suspend() 方法的作用是将一个线程挂起(暂停),
  • resume() 方法的作用则是将一个挂起的线程重新开始并继续向下运行。

通过一个例子来看一下这两个方法的使用:

public class SuspendThread {

    public static void main(String[] args) {
SimpleDateFormat f = new SimpleDateFormat("HH:mm:ss"); try {
MyThread2 t = new MyThread2();
t.start();
Thread.sleep(1000); //暂停线程
t.suspend();
System.out.println("暂停线程:" +
f.format(System.currentTimeMillis()) +
", num = " + t.getNum());
Thread.sleep(5000);
System.out.println("暂停线程5秒后:" +
f.format(System.currentTimeMillis()) +
", num = " + t.getNum()); //继续线程
t.resume();
System.out.println("--恢复线程--");
Thread.sleep(5000);
System.out.println("--恢复线程5秒后--"); //再次暂停线程
t.suspend();
System.out.println("再次暂停:" +
f.format(System.currentTimeMillis()) +
", num = " + t.getNum());
Thread.sleep(5000);
System.out.println("再次暂停5秒后:" +
f.format(System.currentTimeMillis()) +
", num = " + t.getNum()); } catch (Exception e) {
e.printStackTrace();
}
} } class MyThread2 extends Thread{
private long num = 0; public void setNum(long num) {
this.num = num;
} public long getNum(){
return num;
} @Override
public void run() {
while(true){
num++;
}
} }

输出结果:

暂停线程:15:27:21, num = 612874824
暂停线程5秒后:15:27:26, num = 612874824
--恢复线程--
--恢复线程5秒后--
再次暂停:15:27:31, num = 3663392709
再次暂停5秒后:15:27:36, num = 3663392709

从输出结果打印的时间上看来,线程确实被暂停了,而且还能恢复继续运行。

为什么弃用?

虽然现在已经成功的暂停和恢复了线程,但是我们会发现 suspend 和 resume 早已被标注为了废弃方法,并且不推荐使用了。

不释放锁

这是因为 suspend 方法不会释放锁,而如果调用了 suspend 方法的目标线程在挂起时对某一重要的系统资源持有锁,那么在目标线程重新开始之前其他任何线程都无法访问该资源。

public class SynchronizedObject {

    public static void main(String[] args) throws InterruptedException {

        final SynchronizedObject object = new SynchronizedObject();

        Thread thread1 = new MyThread3(object);
Thread thread2 = new MyThread3(object); thread1.setName("thread1");
thread1.start(); Thread.sleep(1000); thread2.setName("thread2");
thread2.start(); } //使用synchronized实现锁功能
synchronized public void printString(){
String threadName = Thread.currentThread().getName(); System.out.println(threadName + " - begin");
if(threadName.equals("thread1")) {
System.out.println("thread1 - suspend");
Thread.currentThread().suspend();
}
System.out.println(threadName + " - end");
} static class MyThread3 extends Thread {
SynchronizedObject obj; public MyThread3(SynchronizedObject obj) {
this.obj = obj;
} @Override
public void run() {
obj.printString();
}
}
}

输出结果:

thread1 - begin
thread1 - suspend

从打印的结果来看,虽然 thread2 被启动了,但是并没有进入它的 printString() 方法,这便是因为 printString 方法已经被 thread1 线程锁定并暂定了,导致其他线程无法访问公共代码块。

不同步

使用 suspend 和 resume 方法也容易出现因为线程的暂停而导致数据不同步的问题。

public class TestObject {

    public static void main(String[] args) throws InterruptedException {
final UserObject obj = new UserObject();
Thread thread1 = new Thread() {
public void run() {
obj.setValue("lisi", "30");
};
}; thread1.setName("thread1");
thread1.start();
thread1.sleep(500); Thread thread2 = new Thread(){
public void run() {
obj.printValue();
};
}; thread2.start(); } static class UserObject {
private String name = "zhangsan";
private String age = "18"; public void setValue(String name, String age) {
this.name = name;
if(Thread.currentThread().getName().equals("thread1")){
System.out.println("暂停 thread1");
Thread.currentThread().suspend();
}
this.age = age;
} private void printValue(){
System.out.println(name + " : " + age);
}
} }

输出结果:

暂停 thread1
lisi : 18

程序运行的结果出现了不同步的情况,所以不再建议使用 suspend 和 resume 方法了。

上一篇:https 学习总结


下一篇:Java中的线程Thread方法之---suspend()和resume()