63.JAVA编程思想——死锁

63.JAVA编程思想——死锁

由于线程可能进入堵塞状态,而且由于对象可能拥有“同步”方法——除非同步锁定被解除,否则线程不能访问那个对象——所以一个线程完全可能等候另一个对象,而另一个对象又在等候下一个对象,以此类推。

这个“等候”链最可怕的情形就是进入封闭状态——最后那个对象等候的是第一个对象!此时,所有线程都会陷入无休止的相互等待状态,大家都动弹不得。我们将这种情况称为“死锁”。尽管这种情况并非经常出现,但一旦碰到,程序的调试将变得异常艰难。

就语言本身来说,尚未直接提供防止死锁的帮助措施,需要我们通过谨慎的设计来避免。如果有谁需要调试一个死锁的程序,他是没有任何窍门可用的。

1. Java 1.2 对stop(),suspend(),resume()以及destroy()的反对为减少出现死锁的可能,Java 1.2 作出的一项贡献是“反对”使用Thread 的stop(),suspend(),resume()以及destroy()方法。之所以反对使用stop(),是因为它不安全。它会解除由线程获取的所有锁定,而且如果对象处于一种不连贯状态(“被破坏”),那么其他线程能在那种状态下检查和修改它们。结果便造成了一种微妙的局面,我们很难检查出真正的问题所在。所以应尽量避免使用stop(),应该采用Blocking.java
那样的方法,用一个标志告诉线程什么时候通过退出自己的run()方法来中止自己的执行。

如果一个线程被堵塞,比如在它等候输入的时候,那么一般都不能象在Blocking.java 中那样轮询一个标志。但在这些情况下,我们仍然不该使用stop(),而应换用由Thread 提供的interrupt()方法,以便中止并退出堵塞的代码。

1     代码

import java.awt.*;

import java.awt.event.*;

import java.applet.*;

class Blocked
extends Thread {

public
synchronizedvoid
run() {

try {

wait();// Blocks

} catch (InterruptedException
e) {

System.out.println("InterruptedException");

}

System.out.println("Exiting run()");

}

}

public
class
Interrupt extends Applet {

private Button
interrupt= newButton("Interrupt");

private Blocked
blocked= newBlocked();

public
void
init() {

add(interrupt);

interrupt.addActionListener(new ActionListener() {

public
void
actionPerformed(ActionEvent
e) {

System.out.println("Button pressed");

if (blocked ==
null)

return;

Threadremove=
blocked;

blocked =
null;
// to releaseit

remove.interrupt();

}

});

blocked.start();

}

public
staticvoid
main(String[]
args){

Interrupt applet =
new Interrupt();

Frame aFrame =
new Frame("Interrupt");

aFrame.addWindowListener(new WindowAdapter() {

public
void
windowClosing(WindowEvent
e) {

System.exit(0);

}

});

aFrame.add(applet, BorderLayout.CENTER);

aFrame.setSize(200, 100);

applet.init();

applet.start();

aFrame.setVisible(true);

}

}/// :~

2     执行

Blocked.run()内部的wait()会产生堵塞的线程。当我们按下按钮以后,blocked(堵塞)的句柄就会设为null,使垃圾收集器能够将其清除,然后调用对象的interrupt()方法。如果是首次按下按钮,我们会看到线程正常退出。但在没有可供“杀死”的线程以后,看到的便只是按钮被按下而已。suspend()和resume() 方法天生容易发生死锁。调用suspend()的时候,目标线程会停下来,但却仍然持有在这之前获得的锁定。此时,其他任何线程都不能访问锁定的资源,除非被“挂起”的线程恢复运行。对任何线程来说,如果它们想恢复目标线程,同时又试图使用任何一个锁定的资源,就会造成令人难堪的死锁。所以我们不应该使用suspend()和resume(),而应在自己的Thread
类中置入一个标志,指出线程应该活动还是挂起。若标志指出线程应该挂起,便用wait()命其进入等待状态。若标志指出线程应当恢复,则用一个notify()重新启动线程。我们可以修改前面的Counter2.java 来实际体验一番。尽管两个版本的效果是差不多的,但大家会注意到代码的组织结构发生了很大的变化——为所有“听众”都使用了匿名的内部类,而且

Thread 是一个内部类。这使得程序的编写稍微方便一些,因为它取消了Counter2.java 中一些额外的记录工作。

3     代码2

import java.awt.*;

import java.awt.event.*;

import java.applet.*;

public
class
Suspend extends Applet {

private TextField
t= newTextField(10);

private Button
suspend= newButton("Suspend"),
resume= newButton("Resume");

class Suspendable
extends Thread {

private
int
count = 0;

private
boolean
suspended =
false;

public Suspendable() {

start();

}

public
void
fauxSuspend() {

suspended =
true;

}

public
synchronized void
fauxResume() {

suspended =
false;

notify();

}

public
void
run() {

while (true) {

try {

sleep(100);

synchronized (this) {

while (suspended)

wait();

}

} catch (InterruptedException
e) {

}

t.setText(Integer.toString(count++));

}

}

}

private Suspendable
ss= newSuspendable();

public
void
init() {

add(t);

suspend.addActionListener(new ActionListener() {

public
void
actionPerformed(ActionEvent
e) {

ss.fauxSuspend();

}

});

add(suspend);

resume.addActionListener(new ActionListener() {

public
void
actionPerformed(ActionEvent
e) {

ss.fauxResume();

}

});

add(resume);

}

public
staticvoid
main(String[]
args){

Suspend applet =
new Suspend();

Frame aFrame =
new Frame("Suspend");

aFrame.addWindowListener(new WindowAdapter() {

public
void
windowClosing(WindowEvent
e) {

System.exit(0);

}

});

aFrame.add(applet, BorderLayout.CENTER);

aFrame.setSize(300, 100);

applet.init();

applet.start();

aFrame.setVisible(true);

}

} /// :~

Suspendable 中的suspended(已挂起)标志用于开关“挂起”或者“暂停”状态。为挂起一个线程,只需调用fauxSuspend()将标志设为true(真)即可。对标志状态的侦测是在run()内进行的。就象早些时候提到的那样,wait()必须设为“同步”(synchronized),使其能够使用对象锁。在fauxResume()中,suspended 标志被设为false(假),并调用notify()——由于这会在一个“同步”从句中唤醒wait(),所以fauxResume()方法也必须同步,使其能在调用notify()之前取得对象锁(这样一来,对象锁可由要唤醍的那个wait()使用)。如果遵照本程序展示的样式,可以避免使用wait()和notify()

Thread 的destroy()方法根本没有实现;它类似一个根本不能恢复的suspend(),所以会发生与suspend()一样的死锁问题。然而,这一方法没有得到明确的“反对”,也许会在Java以后的版本(1.2 版以后)实现,用于一些可以承受死锁危险的特殊场合。

为什么要实现这些现在又被“反对”的方法。之所以会出现这种情况,大概是由于Sun公司主要让技术人员来决定对语言的改动,而不是那些市场销售人员。通常,技术人员比搞销售的更能理解语言的实质。当初犯下了错误以后,也能较为理智地正视它们。这意味着Java 能够继续进步,即便这使Java 程序员多少感到有些不便。就我自己来说,宁愿面对这些不便之处,也不愿看到语言停滞不前。

时间: 2024-12-19 12:37:06

63.JAVA编程思想——死锁的相关文章

64.JAVA编程思想——优先级

64.JAVA编程思想--优先级 线程的优先级(Priority)告诉调试程序该线程的重要程度有多大.如果有大量线程都被堵塞,都在等候运行,调试程序会首先运行具有最高优先级的那个线程.然而,这并不表示优先级较低的线程不会运行(换言之,不会因为存在优先级而导致死锁).若线程的优先级较低,只不过表示它被准许运行的机会小一些而已. 可用getPriority()方法读取一个线程的优先级,并用setPriority()改变它.在下面程序中,大家会发现计数器的计数速度慢了下来,因为它们关联的线程分配了较低

65.JAVA编程思想——关于Runnable

65.JAVA编程思想--关于Runnable 在早些时候,曾建议大家在将一个程序片或主Frame 当作Runnable 的实现形式之前,一定要好好地想一想.若采用那种方式,就只能在自己的程序中使用其中的一个线程.这便限制了灵活性,一旦需要用到属于那种类型的多个线程,就会遇到不必要的麻烦. 当然,如果必须从一个类继承,而且想使类具有线程处理能力,则Runnable 是一种正确的方案.最后一个例子对这一点进行了剖析,制作了一个RunnableCanvas类,用于为自己描绘不同的颜色(Canvas

62.JAVA编程思想——线程堵塞

62.JAVA编程思想--线程堵塞 一个线程可以有四种状态: (1) 新(New):线程对象已经创建,但尚未启动,所以不可运行. (2) 可运行(Runnable ):意味着一旦时间分片机制有空闲的CPU 周期提供给一个线程,那个线程便可立即开始运行.因此,线程可能在.也可能不在运行当中,但一旦条件许可,没有什么能阻止它的运行--它既没有"死"掉,也未被"堵塞". (3) 死(Dead):从自己的run()方法中返回后,一个线程便已"死"掉.亦可

82.JAVA编程思想——关于垃圾收集

82.JAVA编程思想--关于垃圾收集 "很难相信Java 居然能和C++一样快,甚至还能更快一些." 据我自己的实践,这种说法确实成立.然而,我也发现许多关于速度的怀疑都来自一些早期的实现方式.由于这些方式并非特别有效,所以没有一个模型可供参考,不能解释Java 速度快的原因. 之所以想到速度,部分原因是由于C++模型.C++将自己的主要精力放在编译期间"静态"发生的所有事情上,所以程序的运行期版本非常短小和快速.C++也直接建立在C 模型的基础上(主要为了向后兼

《Java编程思想》读书笔记

前言 这个月一直没更新,就是一直在读这本<Java编程思想>,这本书可以在Java业界被传神的一本书,无论谁谈起这本书都说好,不管这个人是否真的读过这本书,都说啊,这本书很好.然后再看这边书的厚度,哇塞,厚的真的不止一点点,所以很多人看了没多久就放弃了,看不下去,但是基于它的厚度,就说,这是一本好书.也有人说,看了没什么用的一本书,甚至还去嘲笑那些正在看的人,说还不如看点实际的技术.那么在我的世界里,如果一本书没有读过,如果妄加评论的话,没有任何的意义.所以我真的仔仔细细读了下来,书上也写了很

异常笔记--java编程思想

开一个新的系列,主要记一些琐碎的重要的知识点,把书读薄才是目的...特点: 代码少,概念多... 1. 基本概念 异常是在当前环境下无法获得必要的信息来解决这个问题,所以就需要从当前环境跳出,就是抛出异常.抛出异常后发生的几件事: 1.在堆上创建异常对象. 2.当前的执行路径中止                                          3. 当前环境抛出异常对象的引用.                                         4. 异常处理机制接

《Java编程思想》第十三章 字符串

<Java编程思想>读书笔记 1.String作为方法的参数时,会复制一份引用,而该引用所指的对象其实一直待在单一的物理位置,从未动过. 2.显式地创建StringBuilder允许预先为他指定大小.如果知道字符串多长,可以预先指定StringBuilder的大小避免多次重新分配的冲突. 1 /** 2 * @author zlz099: 3 * @version 创建时间:2017年9月1日 下午4:03:59 4 */ 5 public class UsingStringBuilder {

Java编程思想 4th 第2章 一切都是对象

Java是基于C++的,但Java是一种更纯粹的面向对象程序设计语言,和C++不同的是,Java只支持面向对象编程,因此Java的编程风格也是纯OOP风格的,即一切都是类,所有事情在类对象中完成. 在Java中,使用引用来操纵对象,在Java编程思想的第四版中,使用的术语是"引用(reference)",之前有读过Java编程思想第三版,在第三版中,使用的术语是"句柄(handle)",事实上,我觉得第三版的术语"句柄"更加形象传神,就像你用一个

《Java编程思想(第4版)》pdf

下载地址:网盘下载 内容简介 编辑 本书赢得了全球程序员的广泛赞誉,即使是最晦涩的概念,在Bruce Eckel的文字亲和力和小而直接的编程示例面前也会化解于无形.从Java的基础语法到最高级特性(深入的面向对象概念.多线程.自动项目构建.单元测试和调试等),本书都能逐步指导你轻松掌握.[1] 从本书获得的各项大奖以及来自世界各地的读者评论中,不难看出这是一本经典之作.本书的作者拥有多年教学经验,对C.C++以及Java语言都有独到.深入的见解,以通俗易懂及小而直接的示例解释了一个个晦涩抽象的概