Java中的线程Thread方法之---suspend()和resume()

2024-06-10 16:08

本文主要是介绍Java中的线程Thread方法之---suspend()和resume(),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

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

package cn.edu.ahui;
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 printCurrentAliveThread(){  Map<Thread, StackTraceElement[]> maps = Thread.getAllStackTraces();  Set<Entry<Thread, StackTraceElement[]>> set = maps.entrySet();  Iterator<Entry<Thread,StackTraceElement[]>> 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("-------------------------");  }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();  }  }  
}  

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

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


好吧,开始打印,打印到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方法注释之后,运行就没有任何问题了。


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

原文地址:http://blog.csdn.net/jiangwei0910410003/article/details/19910517

这篇关于Java中的线程Thread方法之---suspend()和resume()的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1048628

相关文章

JVM 的类初始化机制

前言 当你在 Java 程序中new对象时,有没有考虑过 JVM 是如何把静态的字节码(byte code)转化为运行时对象的呢,这个问题看似简单,但清楚的同学相信也不会太多,这篇文章首先介绍 JVM 类初始化的机制,然后给出几个易出错的实例来分析,帮助大家更好理解这个知识点。 JVM 将字节码转化为运行时对象分为三个阶段,分别是:loading 、Linking、initialization

Spring Security 基于表达式的权限控制

前言 spring security 3.0已经可以使用spring el表达式来控制授权,允许在表达式中使用复杂的布尔逻辑来控制访问的权限。 常见的表达式 Spring Security可用表达式对象的基类是SecurityExpressionRoot。 表达式描述hasRole([role])用户拥有制定的角色时返回true (Spring security默认会带有ROLE_前缀),去

浅析Spring Security认证过程

类图 为了方便理解Spring Security认证流程,特意画了如下的类图,包含相关的核心认证类 概述 核心验证器 AuthenticationManager 该对象提供了认证方法的入口,接收一个Authentiaton对象作为参数; public interface AuthenticationManager {Authentication authenticate(Authenti

Spring Security--Architecture Overview

1 核心组件 这一节主要介绍一些在Spring Security中常见且核心的Java类,它们之间的依赖,构建起了整个框架。想要理解整个架构,最起码得对这些类眼熟。 1.1 SecurityContextHolder SecurityContextHolder用于存储安全上下文(security context)的信息。当前操作的用户是谁,该用户是否已经被认证,他拥有哪些角色权限…这些都被保

Spring Security基于数据库验证流程详解

Spring Security 校验流程图 相关解释说明(认真看哦) AbstractAuthenticationProcessingFilter 抽象类 /*** 调用 #requiresAuthentication(HttpServletRequest, HttpServletResponse) 决定是否需要进行验证操作。* 如果需要验证,则会调用 #attemptAuthentica

Spring Security 从入门到进阶系列教程

Spring Security 入门系列 《保护 Web 应用的安全》 《Spring-Security-入门(一):登录与退出》 《Spring-Security-入门(二):基于数据库验证》 《Spring-Security-入门(三):密码加密》 《Spring-Security-入门(四):自定义-Filter》 《Spring-Security-入门(五):在 Sprin

Java架构师知识体认识

源码分析 常用设计模式 Proxy代理模式Factory工厂模式Singleton单例模式Delegate委派模式Strategy策略模式Prototype原型模式Template模板模式 Spring5 beans 接口实例化代理Bean操作 Context Ioc容器设计原理及高级特性Aop设计原理Factorybean与Beanfactory Transaction 声明式事物

Java进阶13讲__第12讲_1/2

多线程、线程池 1.  线程概念 1.1  什么是线程 1.2  线程的好处 2.   创建线程的三种方式 注意事项 2.1  继承Thread类 2.1.1 认识  2.1.2  编码实现  package cn.hdc.oop10.Thread;import org.slf4j.Logger;import org.slf4j.LoggerFactory

JAVA智听未来一站式有声阅读平台听书系统小程序源码

智听未来,一站式有声阅读平台听书系统 🌟&nbsp;开篇:遇见未来,从“智听”开始 在这个快节奏的时代,你是否渴望在忙碌的间隙,找到一片属于自己的宁静角落?是否梦想着能随时随地,沉浸在知识的海洋,或是故事的奇幻世界里?今天,就让我带你一起探索“智听未来”——这一站式有声阅读平台听书系统,它正悄悄改变着我们的阅读方式,让未来触手可及! 📚&nbsp;第一站:海量资源,应有尽有 走进“智听

【C++】_list常用方法解析及模拟实现

相信自己的力量,只要对自己始终保持信心,尽自己最大努力去完成任何事,就算事情最终结果是失败了,努力了也不留遗憾。💓💓💓 目录   ✨说在前面 🍋知识点一:什么是list? •🌰1.list的定义 •🌰2.list的基本特性 •🌰3.常用接口介绍 🍋知识点二:list常用接口 •🌰1.默认成员函数 🔥构造函数(⭐) 🔥析构函数 •🌰2.list对象