网络编程 | 站长之家 | 网页制作 | 图形图象 | 操作系统 | 冲浪宝典 | 软件教学 | 网络办公 | 邮件系统 | 网络安全 | 认证考试 | 系统进程
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,移动开发
本月文章推荐
.J2EE的发展历程.
.Resin服务器平台介绍简介.
.JRuby:使Java和Ruby成为一家人.
.从Hibernate2到3移植的两点初体验.
.你可以在oracle中发布java程序吗.
.Java中使用正则表达式验证本地化.
.Hibernate为什么成功的.
.Java的中文编程与配置心得.
.2005年Java十大热点问题揭晓.
.Sun复兴2大支柱:x86、Sparc服务.
.常见的十四种Java开发工具及其特.
.J2EE综合应用之——解析MVC模式.
.杂谈:Java编程思想面向对象逻辑.
.Visual Basic串口通讯调试方法.
.JAVA与数据库连接方法(三).
.全面介绍Xen虚拟机 深入学习Xen新.
.Java常用的加密 解密 数字签名等.
.测试概念进行代码设计时的七条基.
.表现层框架Struts/Tapestry/JSF比.
.java入门之:java概述.

突破Java异常处理规则

发表日期:2008-1-5



  Q: 我在我的应用程序中调用了外部方法并且想捕捉它可能抛出的异常。我能捕捉Java.lang.Exception吗?

  A: 通过一个给定的方法去处理所有运行时和检测异常对于预防外部错误是不充分的。

  你可以去读目前 JavaWorld文章 – “Java Tip 134: When Catching Exception, Don’t Cast Your Net Too Wide”。这篇文章警告了捕捉java.lang.Exception和java.lang.Throable是不好的。捕捉你能指定的异常对于代码的可维护性是十分重要的。然而这个规则依靠于非凡的环境。假如你不打算你的程序崩溃并且保留你的数据结构的安全异常,那么你必须捕捉被抛出的真正的异常。


  举个例子,想象你有一个加载了这个接口的服务器应用:

public interface IFoo
{
 /**
 * This method can't throw any checked exceptions...or can it?
 */
 void bar ();
} // End of interface
  对于给出参数的理由是让我们通知你这样的服务在什么地方,并且不同的IFoo实现能够从外部资源加载上。你写如下代码:

try
{
 IFoo foo = ... // get an IFoo implementation
 foo.bar ();
}
catch (RuntimeException ioe)
{
 // Handle 'ioe' ...
}
catch (Error e)
{
 // Handle or re-throw 'e' ...
}
  并且你在这个里处理了所有可能的异常。你不需要在这里加上任何捕捉java.io.IOException的异常,因为IFoo实现没有从IFoo.bar()中抛出它,对吗?(事实上,假如你加上了捕捉java.io.IOException异常块,编译器可能会把它作为不可到达的异常而丢弃)

  错误。在我写的EvilFoo类中bar()方法证实了将抛出你传递给类构造器的任何异常:

public void bar ()
{
 EvilThrow.throwThrowable (m_throwthis);
}
  运行Main方法:

public class Main
{
 public static void main (final String[] args)
 {
  // This try/catch block appears to intercept all exceptions that
  // IFoo.bar() can throw; however, this is not true
  try
  {
   IFoo foo = new EvilFoo (new java.io.IOException ("SURPRISE!"));
   foo.bar ();
  }
  catch (RuntimeException ioe)
  {
   // Ignore ioe
  }
  catch (Error e)
  {
   // Ignore e
  }
 }
} // End of class
  你将看到从bar()方法抛出的java.io.IOException异常实例并且没有任何捕捉块:

>java -cp classes Main
Exception in thread "main" java.io.IOException: SURPRISE!
at Main.main(Main.java:23)
  在这里发生了什么?

  主要的观察是通常针对检测异常的Java规则仅仅在编译的时候被执行。在运行的时候,一个JVM不能保证被一个方法抛出的异常是否和在这个方法中声明的抛出异常相匹配。因为调用方法的职责是捕捉和处理所有从调用方法抛出的异常。任何没有被调用方法声明的异常将不予理睬并且拒绝调用栈。

  假如正常行为是编译器执行,那么我怎么创建EvilFoo的?至少有两个方法可以去创建抛出没有声明的异常的Java方法:

   Thread.stop(Throwable)在一些时候不被赞成使用,但是它仍然被使用并且传递一个Throwable给被调用的Thread。

   分别编译:你能在编译EvilFoo时候不去编译真正声明bar()方法抛出检测异常的IFoo临时版本。

  我用后一种选择:我编译开始定义的EvilThrow类:

public abstract class EvilThrow
{
 public static void throwThrowable (Throwable throwable)
 throws Throwable
 {
  throw throwable;
 }
}
  接下来,我用Byte Code Engineering Library(BCEL)的JasminVisitor分解结果,在汇编代码中删除throwThrowable()方法Throwable的声明,并且用Jasmin assembler 编译新的版本。

  假如你编写捕捉异常的构造器,那么它应该总是捕捉java.lang.Throwable而不仅仅只捕捉java.lang.Exception。这个规则适合你开发治理运行时的应用程序和必须执行可能包含错误甚至恶意代码的外部组件。你要确保捕捉Throwable并且过滤掉错误信息。

  下面示例说明了假如你没有遵循这个建议将发生什么。

Example: Breaking SwingUtilities.invokeAndWait()

  javax.swing.SwingUtilities.invokeAndWait()是在AWT上执行一个线程的有用方法。当一个应用程序线程必须更新图形用户接口并且服从所有Swing线程规则的时候这个方法将被调用。一个没有捕捉Runnable.run()抛出的异常将被捕捉并且被封装在一个InvocationTragetException中重新抛出。

  Sun的J2SE1.4.1假设这样一个未捕捉的异常仅仅是java.lang.Exception的子类。这里是一个SwingUtilities.invokeAndWait()调用java.awt.event.InvocationEvent的一个分析:


上一篇:Jini技术介绍(二) 人气:549
下一篇:Java Applet编程响应鼠标键盘 人气:584
浏览全部Java的内容 Dreamweaver插件下载 网页广告代码 祝你圣诞节快乐 2009年新年快乐