首页 > 其他分享 >异常处理措施

异常处理措施

时间:2023-04-28 16:36:37浏览次数:60  
标签:措施 Java 处理 应用程序 设计 错误处理 异常


异常处理措施

——针对有效的错误处理设计异常管理系统

作者:Jean-Pierre Norguet ,JavaWorld.com,11/15/07

1.            摘要

在面向对象的应用程序中,由于代码重载、错误的问题处理方式,导致异常有越来越多的趋势。在这篇文章中,作者Jean-Pierre Norguet 介绍了如何设计异常,来实现一个简单的、可读的、健壮的、灵活的、面向调试的及用户友好的错误处理系统。在本文中,作者提出了简单异常集合的设计,并且给出了Java 实现的源代码。最后,作者介绍了如何将这样的设计集成到一个Java

在一个面向对象的项目中,设计异常处理的最好途径从来也没有如我们期待的那样清晰。在旧的大的系统中,异常的发生有激增的趋势,最终达到几百行的代码。对于一些常见的编程场景,异常检查是必须的,但是也会带来可观的处理开销。虽然安静的异常捕获可以找到问题的源头,但是避免不了它的一些缺点(虽然这些缺点不是致命的);比如你必须熟悉这些代码。

2.            序

本文将介绍如何通过有限的异常集合来满足一个错误处理的需求。在建立起一个好的错误处理系统框架之后,将会指出异常处理设计中常见的错误,这些错误将会逐渐损害应用程序的性能;然后将会给出一个异常集合的例子,这个例子支持本文讨论的异常处理设计的基本功能:异常系统应该被设计成能够帮助外部系统(用户)处理未知情况(在运行时发生),而不是设计成能够帮助编程人员处理已知情况;还会介绍本文给出例子中的各个类的含义以及如何在一个特定的Java 企业应用程序体系中使用它们;最后给出这个例子的Java

3.            错误处理需求

什么是一个好的错误处理系统?抛开审美角度的考虑,一个好的错误处理系统通常要符合下面的条件:

·任何异常都不会导致应用系统的崩溃。

·在发生异常时,允许应用程序进行相应的处理。

·显示给用户的错误信息要清晰的描述发生了什么错误以及应该采取什么样的处理。

·如果需要辅助信息,错误信息还要帮助用户与帮助部门交互,为帮助部门团队提供必要的信息,是他们能够快速的容易的重现错误。

·日志信息能为开发团队人员在识别错误、在应用程序代码中定位错误产生的位置以及修正错误提供必要的信息。

·错误处理代码不会降低应用程序代码的可读性。必要的时候,错误处理仅仅是一个安全网,它对应用程序的核心功能具有较低的反问权限。

一个错误处理系统的设计符合这些条件才能被认为是完整的。对于大多数Java

3.1.    应该避免的常见用法

可以通过一个有限的异常类集合来满足上面提到的需求。当设计这样的一个异常类集合是,你应该避免一些常见的用法,例如:

·对每个包定义一个异常类集合,这没什么用处而且也会导致系统中异常类的激增。

·对每个异常都提供checked 和non-checked

·最后,抛出和捕获通用异常在很多方面也是错误倾向。

本文下面提到的异常类集合按照错误处理语义分类,避免了异常处理相关的常见问题。尤其是在应用程序的规模和复杂程度增长是,这种方式更值得推荐。

4.            异常处理设计

图1

图1.

在这个图中你会发现有些异常是checked ,而有些是runtime 。为了避免异常抛出声明的开销,checked

1、  告诉调用方法,在这个处理过程中发生了一个可预见的意外情况。在这种情况下,问题的语义已经被处理方法定义了,直接捕获会更好。

2、  告诉外部系统发生了一个未解决的问题,需要根据异常语义来处理这个问题。

4.1.    理解异常语义

在这里阐明本文对异常语义的定义。在作者看来,设计对象的类需要根据它们在现实世界中的等价物。一个水果或者一个人很容易设计成一个Java

作者的设计建议就是异常应该根据它们的目的来设计。系统内部的、自我调整的异常就意味着是帮助系统来处理不可预见的情况。这些异常的目的也就不是模拟系统中的问题,而是给一个系统需要采取什么措施的指示。

5.            一个异常类集合的例子

在图1

1、  BusinessException: 一种异常情况发生。这种情况是可预见的,也可以被调用方法检测到并立即采取措施。

2、  ParameterException: 输入的数据对处理过程不合法。用户被要求重新输入有效数据或者修改处理过程的条件。

3、  TechnicalException: 技术问题,如无效的SQL

4、  CriticalTechnicalException: 技术问题,如数据库崩溃。在这种情况下整个应用程序都ub

这个异常类集合只是一个例子;很多异常类集合都可以参照它来定义。例如,TechnicalException 和CriticalTechnicalException 可以被设计成一个类,这个类声明一个severity

5.1.    异常日志记录

虽然异常语义关注采取的措施,但是出现的问题也很重要。例如,开发团队人员可以使用这些信息调试代码。在异常处理设计中,导致异常的信息能够在以用程序的错误日志中发现。在适当的位置使用一个好的日志记录框架,这个框架能够有效的从异常信息和堆栈跟踪中记录问题信息。

剩下的唯一问题就是怎么样设计异常类使之能够方便的返回信息。一个解决方案就是为异常类声明一个id 属性来代表遇到的问题的种类。另外,问题自身可带有通用异常,也就是把通用异常内嵌到应用程序异常中。在捕获的时候,原始的信息和堆栈跟踪信息能够通过内嵌的异常得到。id

6.            异常处理流程的设计

一旦你已经设计好异常类本身了,下一步需要思考的就是它在应用程序中的处理流程。一个标准的JEE 应用程序体系通常包括四个部分:展现、业务、集成、持久。异常经常在集成和持久部分被抛出。在业务部分,里层的捕获checked 异常,而外层捕获runtime 异常并根据它们的类型来采取相应的处理措施。也可以在业务部分抛出一些checked 异常并且捕获它们。在这种模式下,集成和持久部分,包括业务部分的里层都将runtime 异常转化成具体的处理措施。图2 展示的就是一个典型的JEE

图2. 标准JEE

异常抛出路径是指从持久部分(假设)发生问题到问题被解决所经历的流程。如果持久层的调用方法能够解决这个问题,那么这个异常就直接被捕获并采取相应的处理措施,业务流程一切照常;如果问题不能被解决,异常将内嵌到一个runtime 异常中经过业务部分的中间层传递到应用程序的上层中,在这里,典型的处理方法就是使用一些应用程序控制器来捕获这些runtime 异常并采取相应的处理措施,展现层显示相应的错误信息给用户。直接捕获checked 异常和推迟捕获runtime 异常是异常处理设计中的两种主要方案,如图3

图3. 直接捕获checked 异常和推迟捕获runtime

7.            扩展java.lang.Exception

文中提到的异常处理设计方案在任何的面向对象语言中都可以很容易的实现,包括Java 。一个相似的异常类树已经在标准Java 库中提供了。在这个库中异常被设计为java.lang.Throwable ,ckeched 异常被设计为java.lang.Exception ,runtime 异常被设计为java.lang.RuntimeException

在java.lang.Exception 下,有大量泛语义的业务异常。运行时应用程序异常,如ParamterException 、TechnicalException 、CriticalTechnicalException (见图1 )各自都设计成相应的概要异常,如IllegalArgumentException 、MissingResourceException 、IllegalStateException

在应用程序中,重用Java 标准异常是一个不错的主意,但是由Java 标准类抛出的异常也会导致一些混乱。你可以通过扩展java.lang.Exception 的自定义异常类树来避免这样的混乱。通过自定义的异常类树,你还可以实现内嵌异常和问题ID 。列表1 给出了文中例子的Java 代码实现。注意,它包括内嵌异常和问题ID

列表1. 通过Java 代码实现内嵌异常和问题ID

 

public class NestedException extends RuntimeException {

   protected Exception nestedException;

   protected int issueId;

   public NestedException(String msg, Exception e, int id) {

      super(msg);

      this.nestedException = e;

      this.issueId = id;

   }

   public Exception getNestedException() {

      return this.nestedException;

   }

   public int getIssue() {

      return this.issueId;

   }

}

public interface Issue {

   public final static int UNDEFINED = 0;

   public final static int EXTERNAL_SERVICE_1_DOWN = 1;

   public final static int EXTERNAL_SERVICE_2_DOWN = 2;

   public final static int SQL_STATEMENT_ERROR = 3;

   // ...

}

8.            总结

设计一个满足好的错误处理系统需求的异常类树非常简单。简单的秘诀就是将设计的主要精力集中在系统应该采取什么样的处理措施,而不是关注会出现的什么样的问题。在本文的设计当中,问题的信息封装在异常类里面。通过在处理措施和问题之间分配异常语义,让你将异常类树限制在一个有限的异常类集合中(可能就六七个左右)。这种设计不仅限制了异常类激增,洱海保证代码的可读性,使你在以后的开发中以最佳的清晰度来关注应用程序的业务逻辑的编码。

9.            作者简介

Jean-Pierre Norguet 拥有布鲁塞尔大学的计算机科学和网络工程博士学位。在IBM 经过了三年全职的电子商务应用程序关键任务的Java 开发后,作为团队领导者和教练,他擅长的技术领域包括了整个应用程序开发周期。目前专注于Java 技术、实体集成和迷你Web 应用的研究。除了一些在线Java 文章外,Norguet 博士还与Prentice 、IBM 出版社一起出版了有关JEE 的著作,同时还在ACM 、IEEE 和Springer-Verlag

10.         

本文中提到的异常类集合的完整代码可在 custom exception set

英文原文地址

 

标签:措施,Java,处理,应用程序,设计,错误处理,异常
From: https://blog.51cto.com/u_6174294/6234792

相关文章

  • 灾难处理:卸载无法开机(蓝屏)的exchange服务器
    目的:卸载无法开机(蓝屏)的exchange服务器,从AD层面删除图形化界面卸载: 确保exchange03上没有任何的数据库命令行的模式下进行卸载: ......
  • gateway 异常
    ClassNotFoundException:javax.validation.ValidationException 三、问题nestedexceptionisjava.lang.NoClassDefFoundError:javax/validation/ValidationException原因:项目中没有引用 spring-boot-starter-validation包Asof#19550,WebandWebFluxstartersd......
  • Provisional heads are shown、NullPointerException空指针异常?堆栈与队列的区别?Java
    Provisionalheadsareshown排查是否插件拦截,我的以前没有这种,所以排除本地网络节点问题,连接不到图片服务器,以下是解决方法:1.进入到C盘Windows文件夹System32/drivers/etc目录下,打开hosts文件,绑定下2.改下本地dns为公共dns网络节点导致的问题,一般为运营商导致,产生问题的原因为......
  • 服务器流量异常排查
    执行以下命令,安装iftop工具(iftop工具为Linux服务器下的流量监控小工具)。 yuminstalliftop-y说明如果是Ubuntu系统,请执行 apt-getinstalliftop-y 命令。执行以下命令,安装lsof。 yuminstalllsof-y执行以下命令,运行iftop。如下图......
  • 语音处理加窗分帧
    语音处理加窗分帧一、分帧语音数据和视频数据不同,本没有帧的概念,但是为了传输与存储,我们采集的音频数据都是一段一段的。为了程序能够进行批量处理,会根据指定的长度(时间段或者采样数)进行分段,结构化为我们编程的数据结构,这就是分帧。二、帧移由于我们常用的信号处理方法都要......
  • 面试官:说说你对Java异常的理解
    关注“Java后端技术全栈”回复“000”获取大量电子书背景不管是工作中还是面试中,异常这一块还是非常重要的。作为Java开发人员来说,学会如何处理异常,哪些异常必须自己处理,哪些异常可以往外抛等等,这些都是必需品。生活总有你意想不到的惊喜和意外,我们的code也是一样。异常不懂,佛祖也......
  • C语言处理特定字符串
    C语言处理特定字符串在使用NiosIDE实现串口助手向NiosII系统发送数据时,再将数据发送至FPGA逻辑模块,以此控制LED灯。在串口助手中发送14568936的数据,Nios接收到的数据是形如"14568936\r\n"的字符串,默认以\r\n结尾,要将此字符串转化为四个整型数据。#include<stdio.h>......
  • windows系统处理占用端口号的进程
    遇到占端口被占用,一般通过如下命令处理netstat-ano|findstr端口号杀掉占用端口号的进程taskkill/f/t/im进程号  ......
  • 中断和异常
    中断(interrupt)通常被定义为一个事件,该事件改变处理器执行的指令顺序。中断通常分为同步(synchronous)中断和异步(asynchronous)中断:  同步中断是当指令执行时由CPU控制单元产生的,之所以称为同步,是因为只有在一条指令终止执行后CPU才会发出中断。  异步中断是由其他硬件设备依照......
  • asp.net com,未能转换为类型库。类型库导出程序在处理,时遇到了错误。错误: 找不到元
    我把[assembly:ComVisible(true)]这个设置为true,就报下边的错误错误:程序集“D:\MyDocuments\VisualStudio2005\Projects\ClientOperation\active\bin\Debug\active.dll”未能转换为类型库。类型库导出程序在处理“active.myControl,active”时遇到......