目录
1、编译时异常和运行时异常
1.1 基本概念
1.2 二者区别
1.3 对异常的处理
2、深入try...catch异常
2.1 try...catch的格式
2.2 关于try...catch
2.3 JDK新特性
2.4、上报和捕捉如何选择
3、getMessage()方法和printStackTrace()方法
4、finally子句
5、final、finally和finalize的区别
5.1 final
5.2 finally
5.3 finalize()
6、自定义异常
6.1 自定义异常的步骤
6.2 栈内存程序的改进
7、子类重写的方法抛出编译异常只能更少/小不能更多
编译时异常和运行时异常都是发生在运行阶段,编译阶段异常不会发生;
编译时异常必须在编译(编写)阶段预先处理,如果不处理编译器报错;
所有异常都是在运行阶段发生的,因为只有程序运行阶段才可以new对象。异常的发生就是new异常对象
编译时异常(受检异常、受控异常)发生概率较高,需要在运行之前对其进行预处理;
运行时异常(未受检异常、非受控异常)发生概率较低,运行之前不需要进行预处理;
1)在方法声明的位置上,使用throws关键字;【抛给上一级,谁调用我,我就抛给谁;抛给上一级同样有两种方式】
java中异常发生之后如果一直上抛,最终抛给了main方法,main方法继续向上抛,抛给了调用者JVM,JVM终止程序的执行
2)使用try..cathch语句进行异常的捕捉;【这件事发生了谁也不知道,因为我给抓住了】
public class Sttt{
public static void main(String[] args) {
System.out.println(100/0);
//程序执行到此发生ArithmeticException异常,底层new了一个ArithmeticException异常对象,然后抛给了main方法,main方法最后无法处理,将异常抛给了JVM,JVM最终终止了程序的执行
System.out.println("helloworld");
}
}
//ArithmeticException 继承 RuntimeException,属于运行时异常,在编写程序时不需要对这种异常进行预先处理
public class Sttt{
public static void main(String[] args)throws ClassNotFoundException//处理方式1 {
doSome();//因为doSome方法()的声明位置上有 throws ClassNotFoundException 所以在调用的时候要对这种异常进行预先的处理,不处理,编译器会报错
//Alt + 回车 可以生成
try {
doSome();
} catch (ClassNotFoundException e) {
e.printStackTrace();
}//处理方式2
}
}
public static void doSome() throws ClassNotFoundException{
//ClassNotFoundException类没找到异常,父类是Exception,所以属于编译时异常
}
3)在抛出异常时,可以抛出该异常的父对象
throws后面可以写多个异常,并且用逗号隔开;
一般不建议在main方法上使用throws,因为这个异常如果真的发生了,一定会抛给JVM,JVM只能终止
异常处理机制的作用就是提高程序的健壮性,保证程序出现了异常也能执行,所以main方法中的异常建议是使用try...catch进行捕捉。main不要继续上抛了
注意:只要异常没有捕捉,采用上报的方式,此方法的后续代码不会执行,另外需要注意:try语句块的某一行出现异常,改行后面的代码不会执行,try catch后续的代码仍然执行
try{
//try尝试
m1();
}catch(FileNotFoudException e){
//catch是捕捉异常之后走的分支
System.out.println("文件不存在,可能路径写错了,也可能该文件被删除了");
}
1、catch后面的小括号中的类型可以是具体的异常类型,也可以是该异常类型的父类型
2、catch可以写多个,便于程序的调试,catch写多个的时候从上到下,必须遵守从小到大
try{
}catch(FileNotFoundException|ArithmeticException|NullPointerException e){
}
如果希望调用者来处理,则选择throws上报
public class Sttt{
public static void main(String[] args) {
//这里为了测试两个方法,而new的异常对象,但是没有吧异常对象抛出,JVM认为是一个普通的java对象
NullPointerException e =new NullPointerException("空指针异常!");
//获取异常简单描述信息:这个信息实际上就是构造方法中的String参数
String msg = e.getMessage();
System.out.println(msg);
e.printStackTrace();//打印异常信息,java后台打印异常堆栈信息的时候采用了异步线程的方式打印的
}
}
1)在finally子句中的代码是最后执行的,并且是一定会执行的,即使try语句块的代码出现了异常,
2)finally子句必须和try一起出现,不能单独编写;
3)finally语句通常使用在完成资源的释放/关闭,因为finally语句块中的代码比较有保障,即使try语句块中的代码出现异常,finally中的代码也会正常进行
4)try语句块即使有return,那么finally也会执行,只有当System.exit(0)退出JVM时,才不会执行finally
public class Sttt{
public static void main(String[] args) {
FileInputStream fis = null;//声明位置放到try外面,这样才能在finally中使用
try{
FileInputStream fis = new FileInputStream("D:\\java\javase");
//开始读文件
String s = null;
//这里空指针异常
s.toString();
//流用完需要关闭,因为流是占用资源的
//即使上面程序出现异常,流也必须关系
//放在这里有可能关不了
}catch (FileNotFoundException e){
e.printStackTrace();
}catch (IOException e){
e.printStackTrace();
}catch (NullPointerException e){
e.printStackTrace();
}finally {
//流的关闭放到这里比较保险
//finally中的代码是一定会执行的
//即使try中出现了异常
if(fis != null)//避免空指针异常
try{
//close()方法有异常,采用捕捉的方式
fis.close();
} catch(IOException e){
e.printStackTrace();
}
}
}
}
5)面试题
java的语法规则:
方法体中的代码必须遵循自上而下顺序依次逐行执行(亘古不变的语法)
return语句一旦执行,整个方法必须结束
public class Sttt{
public static void main(String[] args) {
System.out.println(m());//结果是100
}
public static int m(){
int i= 100;
try{
//这行代码出现在 int i = 100;的下面,所以最终结果必须是返回100
//return 语句还必须保证是最后执行的,一旦执行,整个方法结束
return i;
//这里可以理解为自上而下i已经传入了return中但是还没有执行,所以后面的finally无论在return前怎么改变i的值,return i都不会改变
}finally{
i++;
}
}
}
反编译的代码
public static int m{
int i = 100;
int j = i;
i++;
return j;
}
final是一个关键字。表示最终的、不可变的
final int i = 100;
finally也是一个关键字。和try连用,使用在异常处理机制当中
finally语句块中的代码一定会执行的
try{
}finally{
}
finalize()是Object类的一个方法,作为方法名出现,所以finalize是标识符
finalize()方法是JVM的GC垃圾回收器负责调用
第一步:编写一个类继承Exception或者RubtimeException
第二步:写两个构造方法,一个无参构造方法和一个有参构造方法
注意:throw在手动抛异常的时候使用,throws表示上报异常信息给调用者
public class Sttt{
public static void main(String[] args) {
//new了一个异常对象(没有手动抛出)
MyException e = new MyException("用户名不能为空");
//打印异常信息
e.printStackTrace();
//获取异常简单描述信息
String msg = e.getMessage();
System.out.println(msg);
}
}
public class MyException extends Exception{
public MyException(){
}
public MyException(String s){
super(s);
}
}
public class Text {
public static void main(String[] args) {
//创建一个栈对象,初始化容量是10个
Stack s = new Stack();
s.push("12345ty");
s.push(new Object());
s.push(new Object());
s.push(new Object());
s.push(new Object());
s.pop();
s.pop();
s.pop();
s.pop();
s.pop();
s.pop();
//可以使用for循环进行压栈和弹栈
}
}
class Stack{
//存储任何引用类型数据的数组
private Object[] elements;
//有参构造方法
public Stack(Object[] elements) {
this.elements = elements;
}
//无参构造方法
public Stack() {
//一维数组动态初始化
//默认初始化容量为10
this.elements = new Object[10];
}
//栈帧(永远指向栈顶元素)
private int index=-1;
//压栈方法
public void push(Object obj) throws MystackQperationException{
//重点!!!!!!!!!!!!!
if(this.index >= this.elements.length-1){
throw new MystackQperationException("栈内存已满,压栈失败");
//不要进行try...catch,自己new自己抓的操作,必须抛给调用者
}//这里进行了改进
//重点!!!!!!!!!!!!!
index++;
elements[index] = obj;
System.out.println(obj + "元素,压栈成功,栈帧指向" + index);
}
//弹栈方法
public void pop() throws MystackQperationException{
//重点!!!!!!!!!!!!!
if(this.index <= -1) {
//System.out.println("栈内存已空,弹栈栈失败");
throw new MystackQperationException("栈内存已空,弹栈栈失败");
}
//重点!!!!!!!!!!!!!
else
System.out.println(elements[index] + "元素,弹栈成功,栈帧指向" + --index);
}
//自定义栈操作异常
public class MystackQperationException{
public MystackQperationException{
}
public MystackQperationException(String s){
super(s);
}
}
//static实例变量的get方法
public Object[] getElements() {
return elements;
}
//static实例变量的set方法
public void setElements(Object[] elements) {
this.elements = elements;
}
//实例变量栈帧的get方法
public int getIndex() {
return index;
}
//实例变量栈帧的set方法
public void setIndex(int index) {
this.index = index;
}
}
class Animal{
public void doSome(){
}
public void doOther() throws Exception{
}
}
class Cat extends Animal{
public void doSome() throws Exception{
//编译报错
}
public void doOther() throws Exception{
//编译正常
}
public void doOther(){
//编译正常
}
public void doOther() throws NullPointerException{
//编译正常
}
public void doSome() throws RuntimeException{
//运行编译子类可以正常抛出更多,而编译异常不行
}
}