异常:
指的是程序在执行的过程中,出现的非正常的情况,最终会导致Java虚拟机(JVM)非正常停止。
在Java等面向对象的编程语言中,异常本身是一个类,产生异常就是创建异常对象并抛出了一个异常对象。Java处理异常的方式是中断处理。
注意: 异常指的并不是语法错误,语法错了,编译不通过,不会产生字节码文件,根本不能运行。
学习思路整理:
Throwable体系:
Error:严重错误Error,无法通过处理的错误,只能事先避免;
Exception:表示异常,异常产生后程序员可以通过代码的方式纠正,使程序继续运行,是必须要处理的;异常及时我们平时会遇到的。
异常的分类:
异常分为编译异常(非RuntimeException)和非编译异常(RuntimeException);
编译异常:代码无法正确运行,进而无法生成.class的字节码文件,就好比静态方法中不能定义非静态变量...在使用Java开发工具时,会爆红导致编译不通过。
非编译异常:代码能正确运行,能生成对应的字节码文件,但是会在运行的时候报错,就好比我们平时会遇到的空指针异常...
异常处理:
1.把异常的名称,错误原因及异常出现的位置等信息输出在了控制台
2.程序停止执行
非编译异常(运行时异常)的处理方式:
抛异常(throws 异常名,将异常抛出并由java的虚拟机会对相关异常作出处理);
捕获异常(try{可能存在的异常代码块;} catch(异常类 类名){处理方法;})并手动处理;
两种不同的处理方式:
publicclasstest { publicstaticvoidmain(String[] args) { inta=0; intb=1; //除数不能为0,如果除数为0会报异常System.out.println(b/a); } }
我在此处并没有,抛出异常,是底层Java的JVM帮我们判断处理的,Java的JVM会自动帮我们去找到该异常对应的异常类,并抛出异常;
抛异常并没有实际去解决异常,而是有Java的JVM去找到相关异常类并抛出,所以如果后期代码开发的过程中,遇到一些问题无法避免,可以抛出相关异常,让使用者留心,因为如果对接用户的话,会存在很多不确定性;
Java的DateFormat抽象类中的,SimpleDateFormat实现类中的parse()方法就有一个异常需要抛出;
//从给定字符串的开始解析文本以生成日期。该方法可能不会使用给定字符串的整个文本publicDateparse(Stringsource) throwsParseException//ParseException - 如果指定字符串的开头不能被解析。
try ...catch异常:
publicstaticvoidmain(String[] args) { try { inta=0; intb=1; System.out.println(b/a); }catch (ArithmeticExceptione){ System.out.println("注意除数"); } }
thorw关键字:
Java提供了一个throw关键字,它用来 抛出一个指定的异常对象 (可以使自定义的异常对象);如下方法,就不能抛出一个Exception这个宽泛的异常。
指定抛出异常可能出现的问题:(异常不匹配)
throw和throws的区别:
1.throw代表动作,表示抛出一个异常的动作; throws代表一种状态,代表方法可能有异常抛出 ;
2.throw用在方法实现中,而throws用在方法声明中 ;
3.throw只能用于抛出一种指定的异常,而throws可以抛出多个异常;
关键字throws运用于方法声明之上,用于表示当前方法不处理异常,而是提醒该方法的调用者来处理异常(抛出异常)。
自定义异常:
自定义异常就是在我们书写代码时无法避免的错误,在调用的时候给用户提醒的,自定义异常通常要继承自Exception/RuntimeException,在自定的时候可以参考Exception,底层就是继承Throwable。
一环接一环...:
自定义年龄异常类:
//定义Student类publicclassStudent { privateStringname; privateStringsex; privateintage; publicStudent() { } publicStudent(Stringname, Stringsex, intage) { this.name=name; this.sex=sex; this.age=age; } publicStringgetName() { returnname; } publicvoidsetName(Stringname) { this.name=name; } publicStringgetSex() { returnsex; } publicvoidsetSex(Stringsex) { this.sex=sex; } publicintgetAge() { returnage; } publicvoidsetAge(intage) { this.age=age; } }
定义年龄异常类(当年龄小于0,会爆出StudentAgeException异常,所以是运行是异常,继承自RuntimeException类):
publicclassStudentAgeExceptionextendsRuntimeException{ publicStudentAgeException() { super(); } publicStudentAgeException(Stringmess) { super(mess); } }
测试:
importcn.xiaozheng.pojo.Student; publicclassStudentTest { publicstaticvoidmain(String[] args) { Studentstudent=newStudent(); student.setAge(-2); if(student.getAge() <0){ thrownewStudentAgeException("输入的年龄不能小于0"); } System.out.println(student.getAge()); } }
注意:我们现在定义的异常,并不能使用try...catch抛出,因为我们的异常类中并没有定义引发异常抛出的条件,仅仅是为了预防某种已知的异常,所以只能使用相关判断+throw抛出指定的异常。
finally关键字:
在前面final关键字的学习的时候,引申了一个finally关键字,他用在try...catch中,表示无论是否有异常都会输出finally代码块中的内容;
因为异常会引发程序跳转,导致有些语句执行不到。而finally就是解决这个问题的,在finally代码块中存放的代码都是一定会被执行的。
正如上述图片上的异常,异常后的Sout代码并没有执行输出,finally就是为了解决在异常发生后仍要输出的内容就会放在finally块中。
比如后续要学习到的File类,不论是输入流还是输出流,在程序运行后必须要关闭流释放资源的,如果产生异常无法释放资源,导致Java内存溢出就会报错,所以finally是必须的。
fianlly定义格式:
//书写格式:try { 可能出现异常的代码; 。。。} catch(异常类名变量名) { 异常的处理代码; } finally{ 释放资源} //注意事项finally不能单独使用。必须配合着try...catch使用
代码演示:
publicstaticvoidmpublicstaticvoidmain(String[] args) { try { intnum [] = {1,2,3}; //num = null;System.out.println(num[2]); }catch (NullPointerExceptione){ thrownewNullPointerException(); }finally { System.out.println("必须执行我!!!"); } System.out.println("我是try...catch之外的内容"); } intnum [] = {1,2,3}; //num = null; System.out.println(num[2]); }catch (NullPointerException e){ throw new NullPointerException(); }finally { System.out.println("必须执行我!!!"); }System.out.println("我是try...catch之外的内容"); }
情景展示:
如果在try中直接return了,finally还会执行,是不会影响finally输出的。(在前期学习return关键字的时候,我们知道return是结束方法的,所以return之后是没有内容的。)
关于输出的顺序:finally是在return之前输出的,按原有意思解释可知return之后是没有内容输出的,所以return执行必定是最后了。
publicclassDemoFinally { publicstaticvoidmain(String[] args) { System.out.println(newDemoFinally().test()); } //为了看出效果,我把原来的内容提到了方法之中,这样return就可以返回内容,直观的看到效果。publicStringtest(){ try { intnum [] = {1,2,3}; //num = null;System.out.println(num[2]); return"方法执行完毕!"; }catch (NullPointerExceptione){ thrownewNullPointerException(); }finally { System.out.println("必须执行我!!!"); } }