Java异常处理(2)--异常处理机制及自定义异常

简介: Java异常处理(2)--异常处理机制及自定义异常

在编写程序时,经常要在可能出现错误的地方加上检测的代码,如进行x/y运算时,要检测分母为0,数据为空,输入的不是数据而是字符等。过多的if-else分支会导致程序的代码加长、臃肿,可读性差。因此采用异常处理机制。


Java采用的异常处理机制,是将异常处理的程序代码集中在一起,与正常的程序代码分开,使得程序简洁、优雅,并易于维护。


Java提供的是异常处理的抓抛模型。


3ed9eaaeab604d0b82dc7297dbc53edf.png


Java程序的执行过程中如出现异常,会生成一个异常类对象,该异常对象将被提交给Java运行时系统,这个过程称为抛出(throw)异常。


异常对象的生成抛出:

①由虚拟机自动生成:程序运行过程中,虚拟机检测到程序发生了问题,如果在当

前代码中没有找到相应的处理程序,就会在后台自动创建一个对应异常类的实例

对象并抛出——自动抛出

②由开发人员手动创建:Exception exception = new ClassCastException();——创

建好的异常对象不抛出对程序没有任何影响,和创建一个普通对象一样


446ecbf5010e4eaea22550f04cc25c72.png


如果一个方法内抛出异常,该异常对象会被抛给调用者方法中处理。如果异常没有在调用者方法中处理,它继续被抛给这个调用方法的上层方法。这个过程将一直继续下去,直到异常被处理。这一过程称为捕获(catch)异常。如果一个异常回main()方法,并且main()也不处理,则程序运行终止。


Java异常处理的方式


方式一:try-catch-finally

方式二:throws + 异常类型


通过try-catch-finally(finally可选)方式


try{
  //可能出现异常的代码
}catch(异常类型1 变量名e){
  //处理异常的方式1
}catch(异常类型2 变量名e){
  //处理异常的方式2
}
......
finally{
  //一定会执行的代码
}


如果明确知道产生的是何种异常,可以用该异常类作为catch的参数;也可以用其父类作为catch的参数。比 如 : 可 以 用 ArithmeticException 类 作 为 参 数 的 地 方 , 就 可 以 用RuntimeException类作为参数,或者用所有异常的父类Exception类作为参数。但不能是与ArithmeticException类无关的异常。NullPointerException(catch中的语句将不会执行)。


使用try将可能出现异常代码包装起来,在执行过程中,一旦出现异常, 就会生成一个对应异常类的对象,根据此对象的类型,去catch中进行匹配。一旦try中的异常对象匹配到某一个catch时, 就进入catch中进行异常的处理。处理完成,就跳出当前的try-catch结构(在没有写finally的情况)。继续执行其后的代码。


catch中的异常类型如果没有子父类关系,则谁声明在上,谁声明在下无所谓。

catch中的异常类型如果存在子父类关系,则要求子类一定声明在父类的上面,否则会报错。


异常对象处理(一般写在catch语句里面,如e.printStackTrace()):

①返回String的getMessage() ,获取异常信息。

②没有返回值的printStackTrace(),获取异常类名和异常信息,以及异常出现在程序中的位置。


在try结构中声明的变量,再出了try结构以后,就不能再被调用。


try-catch-finally结构可以嵌套。


使用try-catch-finally处理编译时异常,是得程序在编译时就不再报错,但是运行时仍可能报错。相当于我们使用try-catch-finally将一个编译时可能出现的异常,延迟到运行时出现。

开发中,由于运行时异常比较常见,所以我们通常不针对运行时异常编写try-catch-finally。而针对于编译时异常,一定要考虑异常的处理。


通过throws方式


"throws 异常类型"写在方法的声明处,指明此方法执行时,可能会抛出的异常类型。一旦当方法体执行时,出现异常,会在异常代码处生成一个异常类的对象,此对象满足throws后写的异常类型时,就会被抛出。异常代码后续的代码不再执行。


public void readFile(String file) throws FileNotFoundException {
  ......
  // 读文件的操作可能产生FileNotFoundException类型的异常
  FileInputStream fis = new FileInputStream(file);
  ......
}


try-catch-finally真正地将异常给处理掉了。

throws的方式只是将异常抛给了方法的调用者,并没有真正将异常处理掉。


重写方法不能抛出比被重写方法范围更大的异常类型。在多态的情况下,对methodA()方法的调用-异常的捕获按父类声明的异常处理。


public class A {
  public void methodA() throws IOException {
  ......
  }
}
public class B1 extends A {
  public void methodA() throws FileNotFoundException {
  ......
  }
}
public class B2 extends A {
  public void methodA() throws Exception { //报错
  ......
  }
}


如果父类中被重写的方法没有throws方式处理异常,则子类重写的方法也不能使用throws,意味着如果子类重写的方法中有异常,必须使用try-catch-finally方式处理。


手动抛出(生成创建)异常(throw)


Java异常类对象除在程序执行过程中出现异常时由系统自动生成并抛出,也可根据需要使用人工创建并抛出。


首先要生成异常类对象,然后通过throw语句实现抛出操作(提交给Java运行环境)。例如:


IOException e = new IOException();
throw e;


或者直接


throw new IOException();


用户自定义异常类


一般地,用户自定义异常类都是RuntimeException的子类。

自定义异常类必须继承现有的异常类。

自定义异常类通常需要编写几个重载的构造器。

自定义异常需要提供serialVersionUID。

自定义的异常通过throw抛出。


例子


class MyException extends Exception {
  static final long serialVersionUID = 13465653435L;
  private int idnumber;
  public MyException(String message, int id) {
  super(message);
  this.idnumber = id;
  }
  public int getId() {
  return idnumber;
  }
}


public class MyExpTest {
  public void regist(int num) throws MyException {
  if (num < 0)
    throw new MyException("人数为负值,不合理", 3);
  else
    System.out.println("登记人数" + num);
  }
  public void manager() {
  try {
    regist(100);
  } catch (MyException e) {
    System.out.print("登记失败,出错种类" + e.getId());
  }
  System.out.print("本次登记操作结束");
  }
public static void main(String args[]) {
  MyExpTest t = new MyExpTest();
  t.manager();
  }
}


相关文章
|
29天前
|
Java
java实现从HDFS上下载文件及文件夹的功能,以流形式输出,便于用户自定义保存任何路径下
java实现从HDFS上下载文件及文件夹的功能,以流形式输出,便于用户自定义保存任何路径下
87 34
|
1月前
|
Java
Java 异常处理:11 个异常处理最佳实践
本文深入探讨了Java异常处理的最佳实践,包括早抛出晚捕获、只捕获可处理异常、不忽略异常、抛出具体异常、正确包装异常、记录或抛出异常但不同时执行、不在finally中抛出异常、避免用异常控制流程、使用模板方法减少重复代码、抛出与方法相关的异常及异常处理后清理资源等内容,旨在提升代码质量和可维护性。
|
1月前
|
Java 程序员
深入理解Java异常处理机制
Java的异常处理是编程中的一块基石,它不仅保障了代码的健壮性,还提升了程序的可读性和可维护性。本文将深入浅出地探讨Java异常处理的核心概念、分类、处理策略以及最佳实践,旨在帮助读者建立正确的异常处理观念,提升编程效率和质量。
129 1
|
1月前
|
Java 开发者 UED
深入探索Java中的异常处理机制##
本文将带你深入了解Java语言中的异常处理机制,包括异常的分类、异常的捕获与处理、自定义异常的创建以及最佳实践。通过具体实例和代码演示,帮助你更好地理解和运用Java中的异常处理,提高程序的健壮性和可维护性。 ##
54 2
|
1月前
|
Java API 开发者
深入理解Java中的异常处理机制
本文探讨了Java编程语言中异常处理的核心概念,包括异常类型、异常捕获与抛出、以及最佳实践。通过分析常见的异常场景和处理策略,旨在帮助开发者更好地理解和运用异常处理机制,提高代码的健壮性和可维护性。文章不仅涵盖了基本的try-catch结构,还深入讨论了自定义异常的创建与使用,以及finally块的重要性和应用。此外,还将介绍一些高级技巧,如多异常捕获和嵌套异常处理,为读者提供全面的技术指导。
97 0
|
3天前
|
监控 Java
java异步判断线程池所有任务是否执行完
通过上述步骤,您可以在Java中实现异步判断线程池所有任务是否执行完毕。这种方法使用了 `CompletionService`来监控任务的完成情况,并通过一个独立线程异步检查所有任务的执行状态。这种设计不仅简洁高效,还能确保在大量任务处理时程序的稳定性和可维护性。希望本文能为您的开发工作提供实用的指导和帮助。
39 17
|
13天前
|
Java
Java—多线程实现生产消费者
本文介绍了多线程实现生产消费者模式的三个版本。Version1包含四个类:`Producer`(生产者)、`Consumer`(消费者)、`Resource`(公共资源)和`TestMain`(测试类)。通过`synchronized`和`wait/notify`机制控制线程同步,但存在多个生产者或消费者时可能出现多次生产和消费的问题。 Version2将`if`改为`while`,解决了多次生产和消费的问题,但仍可能因`notify()`随机唤醒线程而导致死锁。因此,引入了`notifyAll()`来唤醒所有等待线程,但这会带来性能问题。
Java—多线程实现生产消费者
|
15天前
|
安全 Java Kotlin
Java多线程——synchronized、volatile 保障可见性
Java多线程中,`synchronized` 和 `volatile` 关键字用于保障可见性。`synchronized` 保证原子性、可见性和有序性,通过锁机制确保线程安全;`volatile` 仅保证可见性和有序性,不保证原子性。代码示例展示了如何使用 `synchronized` 和 `volatile` 解决主线程无法感知子线程修改共享变量的问题。总结:`volatile` 确保不同线程对共享变量操作的可见性,使一个线程修改后,其他线程能立即看到最新值。
|
15天前
|
消息中间件 缓存 安全
Java多线程是什么
Java多线程简介:本文介绍了Java中常见的线程池类型,包括`newCachedThreadPool`(适用于短期异步任务)、`newFixedThreadPool`(适用于固定数量的长期任务)、`newScheduledThreadPool`(支持定时和周期性任务)以及`newSingleThreadExecutor`(保证任务顺序执行)。同时,文章还讲解了Java中的锁机制,如`synchronized`关键字、CAS操作及其实现方式,并详细描述了可重入锁`ReentrantLock`和读写锁`ReadWriteLock`的工作原理与应用场景。
|
16天前
|
安全 Java 编译器
深入理解Java中synchronized三种使用方式:助您写出线程安全的代码
`synchronized` 是 Java 中的关键字,用于实现线程同步,确保多个线程互斥访问共享资源。它通过内置的监视器锁机制,防止多个线程同时执行被 `synchronized` 修饰的方法或代码块。`synchronized` 可以修饰非静态方法、静态方法和代码块,分别锁定实例对象、类对象或指定的对象。其底层原理基于 JVM 的指令和对象的监视器,JDK 1.6 后引入了偏向锁、轻量级锁等优化措施,提高了性能。
42 3