zoukankan      html  css  js  c++  java
  • 使用 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 方法了。

  • 相关阅读:
    解决百度云大文件不能被其他下载器下载
    python中IndentationError: expected an indented block错误的解决方法
    2017Windows下安装pip
    α-β剪枝算法的java语言实现(非常实用)
    flask+sqlite3+echarts2+ajax数据可视化
    flask+sqlite3+echarts2+ajax数据可视化报错:UnicodeDecodeError: 'utf8' codec can't decode byte解决方法
    Windows下快速安装Flask的一次经历
    决策树ID3算法的java实现(基本适用所有的ID3)
    继续上篇抢QQ口令红包,抢那招抢不了的红包技巧
    【QQ红包】手机发抢不到的口令红包
  • 原文地址:https://www.cnblogs.com/liyutian/p/10312767.html
Copyright © 2011-2022 走看看