动态网站制作指南 [  QQ表情  ]
[ 投票调查 ]
[ 企业邮箱 ]
[ 网站空间 ]
网络编程 | 站长之家 | 网页制作 | 图形图象 | 操作系统 | 冲浪宝典 | 软件教学 | 网络办公 | 邮件系统 | 网络安全 | 认证考试 | 系统进程
ASP源码 | .Net源码 | PHP源码 | JSP源码 | JAVA源码 | CGI源码 | VB源码 | C++源码 | Delphi源码 | PB源码 | VF源码 | 汇编 | 服务器
Firefox | IE | Maxthon | 迅雷 | 电驴 | BitComet | FlashGet | QQ | QQ空间 | Vista | 输入法 | Ghost | Word | Excel | wps | Powerpoint
asp | .net | php | jsp | Sql | c# | Ajax | xml | Dreamweaver | FrontPages | Javascript | css | photoshop | fireworks | Flash | Cad | Discuz!
当前位置 > 网站建设学院 > 网络编程 > Java教程
Tag:注入,存储过程,分页,安全,优化,xmlhttp,fso,jmail,application,session,防盗链,stream,无组件,组件,md5,乱码,缓存,加密,验证码,算法,cookies,ubb,正则表达式,水印,索引,日志,压缩,base64,url重写,上传,控件,Web.config,JDBC,函数,内存,PDF,迁移,结构,破解,编译,配置,进程,分词,IIS,Apache,Tomcat,phpmyadmin,Gzip,触发器,socket
网络编程:ASP教程,ASP.NET教程,PHP教程,JSP教程,C#教程,数据库,XML教程,Ajax,Java,Perl,Shell,VB教程,Delphi,C/C++教程,软件工程,J2EE/J2ME,移动开发
文章搜索服务
邮件订阅
输入你的邮件地址,
你将不会错过任何关于:
[ Java教程 ]的信息

本月文章推荐
.使用Socket连接穿越CMWAP代理.
.Struts中不同的Action和ActionFo.
.Java违例控制.
.Hibernate配置文件中映射元素详解.
.类注释文档编写方法.
.将 Java Web 应用从 Windows 移植.
.使用JSR-184里的Sprite3D对象.
.Mini Java编译器(一).
.Java 数据对象(JDO)介绍(四).
.系统编程 在Java中处理日志记录.
.Java初学者入门经典:面向对象的.
.谈谈JAVA中的调用方式.
.使用SOAP开发Java Web服务详解.
..net 的 MSMQ 异步调用全过程分析.
.Hibernate的一对一关系(转).
.ANT十五大最佳实践.
.建立Java的应用程序.
.从XML到Java代码的数据绑定之二.
.用Java实现数据库应用系统(1).
.Think in java 3rd 中文版2.

死锁

发表日期:2008-1-5 |



  由于线程可能进入堵塞状态,而且由于对象可能拥有“同步”方法——除非同步锁定被解除,否则线程不能访问那个对象——所以一个线程完全可能等候另一个对象,而另一个对象又在等候下一个对象,以此类推。这个“等候”链最可怕的情形就是进入封闭状态——最后那个对象等候的是第一个对象!此时,所有线程都会陷入无休止的相互等待状态,大家都动弹不得。我们将这种情况称为“死锁”。尽管这种情况并非经常出现,但一旦碰到,程序的调试将变得异常艰难。
  就语言本身来说,尚未直接提供防止死锁的帮助措施,需要我们通过谨慎的设计来避免。假如有谁需要调试一个死锁的程序,他是没有任何窍门可用的。
  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()方法,以便中止并退出堵塞的代码。
  //: Interrupt.java
  // The alternative approach to using stop()
  // when a thread is blocked
  import java.awt.*;
  import java.awt.event.*;
  import java.applet.*;
  class Blocked extends Thread {
   public synchronized void run() {
   try {
   wait(); // Blocks
   } catch(InterruptedException e) {
   System.out.println("InterruptedException");
   }
   System.out.println("Exiting run()");
   }
  }
  public class Interrupt extends Applet {
   private Button
   interrupt = new Button("Interrupt");
   private Blocked blocked = new Blocked();
   public void init() {
   add(interrupt);
  interrupt.addActionListener(
   new ActionListener() {
   public
   void actionPerformed(ActionEvent e) {
   System.out.println("Button pressed");
   if(blocked == null) return;
   Thread remove = blocked;
   blocked = null; // to release it
   remove.interrupt();
   }
   });
   blocked.start();
   }
   public static void 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);
   }
  } ///:~
  Blocked.run()内部的wait()会产生堵塞的线程。当我们按下按钮以后,blocked(堵塞)的句柄就会设为null,使垃圾收集器能够将其清除,然后调用对象的interrupt()方法。假如是首次按下按钮,我们会看到线程正常退出。但在没有可供“杀死”的线程以后,看到的便只是按钮被按下而已。
  suspend()和resume()方法天生轻易发生死锁。调用suspend()的时候,目标线程会停下来,但却仍然持有在这之前获得的锁定。此时,其他任何线程都不能访问锁定的资源,除非被“挂起”的线程恢复运行。对任何线程来说,假如它们想恢复目标线程,同时又试图使用任何一个锁定的资源,就会造成令人难堪的死锁。所以我们不应该使用suspend()和resume(),而应在自己的Thread类中置入一个标志,指出线程应该活动还是挂起。若标志指出线程应该挂起,便用wait()命其进入等待状态。若标志指出线程应当恢复,则用一个notify()重新启动线程。我们可以修改前面的Counter2.java来实际体验一番。尽管两个版本的效果是差不多的,但大家会注重到代码的组织结构发生了很大的变化——为所有“听众”都使用了匿名的内部类,而且Thread是一个内部类。这使得程序的编写稍微方便一些,因为它取消了Counter2.java中一些额外的记录工作。
  //: Suspend.java
  // The alternative approach to using suspend()
  // and resume(), which have been deprecated
  // in Java 1.2.
  import java.awt.*;
  import java.awt.event.*;
  import java.applet.*;
  
  public class Suspend extends Applet {
   private TextField t = new TextField(10);
   private Button
   suspend = new Button("Suspend"),
   resume = new Button("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 = new Suspendable();
   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 static void 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程序员多少感到有些不便。就我自己来说,宁愿面对这些不便之处,也不愿看到语言停滞不前。
上一篇:JAVA调试技术 人气:681
下一篇:资源访问的错误方法 人气:430
浏览全部Java的内容 Dreamweaver插件下载 常用网页广告代码全集
  最新网站源码 最新软件下载
2008-12-1 MyBB v1.4.4 简体中文版 bulid 2
2008-12-1 新云网站内容管理系统 v4.0.0.11
2008-12-1 网趣网上购物系统时尚版 v8.8
2008-12-1 Textpattern v4.0.7 多国语言版
2008-12-1 Piwik ( PHP统计系统,可以和GOOG
2008-12-1 天空网络电影系统SKYUC! v2.6.2
2008-12-1 SiteDynamic企业网站管理系统 v1
2008-12-1 KindEditor HTML在线编辑器 v3.0
2008-12-1 0451sky高校教务管理系统2008 v4
2008-11-29 Tencent Traveler 4.4
2008-11-29 龙卷风网络收音机 v3.0.0.0
2008-11-29 Intel Chipset Software Install
2008-11-29 TweakVI 1.0 Build 1100
2008-11-29 Opera 9.62 Build 10469
2008-11-29 MPlayer WW编译版 SVN-r28044(20
2008-11-29 NetTools网络工具v1.0.0破解版
2008-11-29 3DGallery三维体验1.1破解版
2008-11-29 SecretBook保密本v1.0破解版
  发表评论
姓 名: 验证码:
内 容:
站长工具:网站收录查询 | Google PR查询 | ALEXA排名查询 | CSS在线编辑器 | OPEN参数生成器 | 弹出式窗口代码产生器 | 密码登录生成器 | 在线按钮生成器 | Meta标签生成器 | 多色彩特效字代码生成器 | 网页代码调试器 | 在线FTP登陆 | Flash取色器 | 配色代码对照表 | 配色辞典 | CSS生成器 | 广告代码 | 框架网页代码生成器 | js/vbs加密 | md5加密 | 进制转换 | UTF-8 转换工具 | 在线调色板 | Html转换js | Html转换asp | Html转换php | Html转换perl
实用工具:汉字翻译拼音 | 拼音字典 | 符号对照表 | 个税计算 | 实时汇率查询换算 | 经典小工具 | 汉字简繁转换 | 普通单位换算 | 公制单位换算 | 生辰老黄历 | 国内电话区号 | 国家代码与域名缩写 | 文字加密解密 | 元素周期表 | 健康查询 | 世界时间 | 万年历 | 二十四节气 | 汉字横竖排版 | 手机号码查询 | 计算器 | ip搜索
业务联系 | 广告刊登 | 频道合作 | 投稿荐稿 | 联系方式 | 加入收藏 | RSS订阅
Copyright © 2000-2009 www.knowsky.com All rights reserved | 沪ICP备05001343号
ホームページ制作 不動産検索システム 求人情報
防水工事·改修工事 フットサル大会 探偵
SEO対策 中国語教室 ホームページ作成