什么是异常
- 概念:程序在运行过程中出现的不正常现象。出现异常不处理将终止程序运行。
- 异常处理的必要性:任何程序都可能存在大量的未知问题、错误;如果不对这些问题进行正确处理,则可能导致程序中断,造成不必要的损失。
- 异常处理:Java编程语言使用异常处理机制为程序提供了异常处理的能力。
异常的分类
Throw able:可抛出的,一切错误或异常的父类位于java.lang包中。
Error:Jvm、硬件、执行逻辑错误,不能手动处理。
StackOverflowError
OutOfMemoryError
Exception:程序在运行和配置中产生的问题,可处理。
RuntimeException:运行时异常,可处理,可不处理。
checkedException:检查时异常,必须处理。
异常的产生
- 当程序在运行时遇到不符合规范的代码或结果时,会产生异常或程序员使用throw关键字手动抛出异常。
异常的传递
- 异常的传递:按照方法的调用链反向传递,如始终没有处理异常,最终会由JVM进行默认异常处理(打印堆栈跟踪信息)。
异常处理
- JAVA的异常处理是通过5个关键字来实现的:
- try:执行可能产生异常的代码
- catch:捕获异常,并处理
- finally:无论是否发生异常,代码总能执行
- throw:手动抛出异常
- throws:声明方法可能要抛出的各种异常
异常的传递
- 异常的传递:按照方法的调用链反向传递,如始终没有处理异常,最终会由JVM进行默认异常处理(打印堆栈跟踪信息)。
异常处理
try{
//可能出现异常的代码
}catch(Exception e){
//捕获异常,并处理
e.printStackTrace();
}finally{
// 释放资源
}
- 无论是否发生异常finally都会执行
- 但有一种情况不会执行就是当退出JVM虚拟机时(System.exit()😉。
try…finally不能捕获异常,仅仅用来当发生异常时,用来释放资源。
一般用在底层代码,只释放资源不做异常处理,把异常向上抛出。
try{
//可能发生异常的代码
}finally{
//是否发生异常都会执行,可以释放资源等...
}
package com.xu.c1;
import java.util.Scanner;
/**
* 演示异常的产生和传递 要求:输入两个数字实现两个数字扣除
*
* @author 【徐先生】
*
*/
public class Demo06 {
public static void main(String[] args) {
try {
divide();
} catch (Exception e) {
System.out.println("出现异常:/by zero");
}
}
public static void divide() {
Scanner input = new Scanner(System.in);
System.out.println("请输入第一个数字:");
int result = 0;
try {
int num1 = input.nextInt();
System.out.println("请输入第二个数字:");
int num2 = input.nextInt();
result = num1 / num2;
} finally {
System.out.println("释放资源");
}
System.out.println("结果:" + result);
System.out.println("程序结束:");
}
}
声明异常
- 如果在一个方法体中抛出了异常,如何通知调用者?
- throws关键字:声明异常
- 使用原则:底层代码向上声明或者抛出异常,最上层一定要处理异常,否则程序中断。
package com.xu.c1;
import java.util.Scanner;
public class Demo06 {
public static void main(String[] args) throws Exception{
divide();//最好在上层处理了,不要交给JVM,否则程序就中断了
}
public static void divide() throws Exception{
Scanner input = new Scanner(System.in);
System.out.println("请输入第一个数字:");
int result = 0;
try {
int num1 = input.nextInt();
System.out.println("请输入第二个数字:");
int num2 = input.nextInt();
result = num1 / num2;
} finally {
System.out.println("释放资源");
}
System.out.println("结果:" + result);
System.out.println("程序结束:");
}
}
抛出异常
- 除了系统自动抛出异常外,有些问题需要程序员自行抛出异常。
- throw关键字:抛出异常
- 语法:throw 异常对象;
自定义异常
- 需继承自Exception或Exceptiond的子类,常用RuntimeException。
- 必要提供的构造方法:
- 无参数构造方法
- String message参数构造方法
package com.xu.c1;
/**
* 自定义异常类
* (1)继承Exception或子类
* (2)添加构造方法
* @author 【徐先生】
*
*/
public class AgeException extends RuntimeException{
public AgeException() {
// TODO Auto-generated constructor stub
}
public AgeException(String message, Throwable cause, boolean enableSuppression, boolean writableStackTrace) {
super(message, cause, enableSuppression, writableStackTrace);
// TODO Auto-generated constructor stub
}
public AgeException(String message, Throwable cause) {
super(message, cause);
// TODO Auto-generated constructor stub
}
public AgeException(String message) {
super(message);
// TODO Auto-generated constructor stub
}
public AgeException(Throwable cause) {
super(cause);
// TODO Auto-generated constructor stub
}
}
package com.xu.c1;
public class SexException extends RuntimeException{
public SexException() {
// TODO Auto-generated constructor stub
}
public SexException(String message, Throwable cause, boolean enableSuppression, boolean writableStackTrace) {
super(message, cause, enableSuppression, writableStackTrace);
// TODO Auto-generated constructor stub
}
public SexException(String message, Throwable cause) {
super(message, cause);
// TODO Auto-generated constructor stub
}
public SexException(String message) {
super(message);
// TODO Auto-generated constructor stub
}
public SexException(Throwable cause) {
super(cause);
// TODO Auto-generated constructor stub
}
}
方法覆盖(重写)
- 带有异常声明的方法覆盖:
- 方法名、参数列表、返回值类型必须和父类相同。
- 子类的访问修饰符和父类相同或是比父类更宽。
- 子类中的方法,不能抛出比父类更多、更宽的检查时异常。(Runtime Exception无限制)