Java线程停止、暂停和恢复等状态的控制

举报
junehappylove1 发表于 2020/12/29 10:36:09 2020/12/29
【摘要】 JDK1.0定义了stop和suspend方法,stop用来直接终止线程,suspend会阻塞线程直到另一个线程调用resume.stop和suspend都有一些共同的点:都试图专横的控制一个给定了的线程的行为。从JDK1.2开始,这两个方法都被弃用了.stop天生就不安全,而经验告诉我们呢suspend方法会经常导致死锁。网上有很多关于Java线程停止、暂停的文章,经过自己测试,在这里总结...

JDK1.0定义了stop和suspend方法,stop用来直接终止线程,suspend会阻塞线程直到另一个线程调用resume.
stop和suspend都有一些共同的点:都试图专横的控制一个给定了的线程的行为。

从JDK1.2开始,这两个方法都被弃用了.stop天生就不安全,而经验告诉我们呢suspend方法会经常导致死锁。
网上有很多关于Java线程停止、暂停的文章,经过自己测试,在这里总结一下。针对不同使用场景选择合适的方法。

1、线程停止

正常情况下线程在执行完run方法之后就会停止,而且不会再恢复,但是我们可能会遇到这样的场景,在线程执行到某一时刻的时候就想立刻停止线程,
如果在run方法中只是处理一些逻辑,而没有sleep、wait等会导致线程阻塞的方法时,最好的办法是使用标识符讲线程停止

public class MyThread extends Thread {
    private volatile boolean stopFlag = false;

    @Override
    public void run() {
        while (!stopFlag) {
            //doSomething

        }


    }
    public boolean isStopFlag() {
        return stopFlag;
    }
    public void setStopFlag(boolean stopFlag) {
        this.stopFlag = stopFlag;
    }
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19

这里需要注意的是stopFlag 要用volatile 修饰,可以确保同一时刻只有同一个线程访问,具体volatile 修饰符的详解参阅《Java理论与实践:正确使用 Volatile 变量》

当run方法中有sleep等方法时,上边这种办法就不能立刻的将线程停止,就需要用到thread.interrupt()。

@Override
    public void run() {
        while (!stopFlag) {
            //doSomething
            try {
                Thread.sleep(100);
            } catch (InterruptedException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
                break;
            }
        }


    }
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15

当调用thread.interrupt()时会立马停止Thread.sleep(100)。并抛出异常,这个时候在catch语句中加入break跳出循环体即可结束线程。
这里需要注意的是interrupt这个方法的作用只是抛出一个异常,并不会终止线程。

2、线程暂停

这里用到Object的wait()方法和notify()和notifyAll()方法。先看代码

public class MyThread extends Thread {
    private String control = "";//只是任意的实例化一个对象而已

    private boolean suspend = false;//线程暂停标识

    @Override
    public void run() {
        while (true) {
            synchronized (control) {
                if (suspend) {
                    try {
                        control.wait();
                    } catch (InterruptedException e) {
                        // TODO Auto-generated catch block
                        e.printStackTrace();
                    }
                }
            }

            //doSomething
        }


    }

    public void setSuspend(boolean suspend) {
        if (!suspend) {
            synchronized (control) {
                control.notifyAll();
            }
        }
        this.suspend = suspend;
    }
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34

这里提供一个public方法setSuspend来设置暂停标识。如果为true时当前线程获取control对象的锁,并执行wait()方法。此时线程会一直处于阻塞状态,直到control对象
调用notify或者notifyAll方法。

【版权声明】本文为华为云社区用户原创内容,转载时必须标注文章的来源(华为云社区)、文章链接、文章作者等基本信息, 否则作者和本社区有权追究责任。如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。