java thread resume_Java中的线程Thread方法之---suspend()和resume()

前篇说到了Thread中的join方法,这一篇我们就来介绍一下suspend()和resume()方法,从字面意义上可以了解到这两个方法是一对的,suspend()方法就是将一个线程挂起(暂停),resume()方法就是将一个挂起线程复活继续执行。首先看一个例子:

package com.threadstop.demo;

import java.util.Iterator;

import java.util.Map;

import java.util.Map.Entry;

import java.util.Set;

/**

* @author weijiang204321

*说明:

*调用了suspend方法,将t0线程挂起,但是出现的问题是,t0.suspend方法之后的代码不执行了,搞了半天终于知道为什么了,

*因为在t0里面使用了System.out.println方法了,查看println方法的源代码发现他是个同步的方法,加锁了,这个锁是哪个呢?

*对就是PrintStream,在Main中的printCurrentAliveThread方法中用到了System.out.println方法,打断点才知道

*搞了半天阻塞在这里了,因为我们知道suspend方法是不释放锁的,所以导致会阻塞在println方法中,但是有一个前提是t0线程和main线程

*的println方法中拿到的是同一个锁,这时候在看一下System.out变量时一个static PrintStream,这时候就明了了,因为是static

*所以对象是相同的,这两个线程拿到的System.out是同一个对象,所以这两个线程也是拿到的是相同的锁的。

*

*/

public class ThreadStopLock {

public static void main(String[] args) {

try {

//定义线程

Thread t0 = new Thread() {

public void run() {

try {

for(long i=0;i<1000*1000*10;i++){

System.out.println(i);

}

System.out.println("thread death");

} catch (Throwable ex) {

System.out.println("Caught in run: " + ex);

ex.printStackTrace();

}

}

};

//开启线程

t0.start();

//等待2s之后挂起线程t0

Thread.sleep(2*1000);

//挂起线程

t0.suspend();

//打印当前的所有线程

printCurrentAliveThread();

//等待2s之后恢复线程

Thread.sleep(2*1000);

//复活线程

t0.resume();

} catch (Throwable t) {

System.out.println("Caught in main: " + t);

t.printStackTrace();

}

}

/**

* 打印当前线程

*/

public static void printCurrentAliveThread(){

Map maps = Thread.getAllStackTraces();

Set> set = maps.entrySet();

Iterator> iterator = set.iterator();

System.out.println("System Alive Thread List:");

System.out.println("-------------------------");

while(iterator.hasNext()){

System.out.println("AliveThread_Name:"+iterator.next().getKey().getName());

}

System.out.println("-------------------------");

}

}代码很简单的,定义一个线程,在线程中进行打印,在主线程中的逻辑是,先开启线程t0进行打印数据,等待2s之后将挂起线程t0,然后打印一下当前的活跃线程,然后再等待2s之后再复活t0线程继续打印!

但是结果不尽人意呀,结果很是惊讶的!运行结果:

58b37c70567f7af8d1aee1ace52b3a19.png

好吧,开始打印,打印到311800(当然这个不是一定的),就停止了,但是这一停止不是停2s呀,是一直停着,这不是明显的死锁吗?导致t0.suspend后面的代码都不执行了,这就郁闷了,这个是为什么呢?纠结了一下午,打断点的时候才知道为什么,断点停在方法printCurrentAliveThread中的System.out.println()这行代码上,那就可以断定了,发生死锁的原因可能就是System.out.println方法,查看源代码:

public void println(String x) {

synchronized (this) {

print(x);

newLine();

}

}这个是PrintStream对象中的println方法,是个同步锁的方法,这时候应该就明白了,那么这个锁是什么呢?没错,这个所就是PrintStream对象,因为在t0线程中也有System.out.println,当调用suspend()方法调用的时候,这个方法是不会释放锁的,当然这个锁是同一个的,为什么呢?看一下System类中的out变量定义:

public final static PrintStream out = null;是static类型的,所以他是类上面的锁,肯定是同一个锁了,所以发生了死锁,

这时候我们将printCurrentAliveThread方法注释之后,运行就没有任何问题了。

就写到这里吧,好累,后面还有哦,继续关注吧!

分享到:

18e900b8666ce6f233d25ec02f95ee59.png

72dd548719f0ace4d5f9bca64e1d7715.png

2014-02-25 14:37

浏览 177

评论

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值