您的位置:首页 > 编程语言 > Java开发

JavaSE 异常

2016-09-13 11:26 555 查看
异常:就是程序运行时出现不正常的情况。

异常由来:问题也是现实生活中的一个具体食物,也可以通过JAVA的类的形式进行描述,并封装成对象。

其实就是JAVA对不正常情况进行描述后的对象体现。

对于问题的划分:两种:一种严重的问题,一种非严重的问题。

对于严重的,JAVA通过Error类进行描述。

对于Error一般不编写针对性的代码对其惊醒处理。

对于非严重的,Java通过Exception类进行描述。

对于Exception可以使用针对性的处理方式进行处理。

异常的处理

Java 提供了特有的语句进行处理

try…catch…finally

try{
需要被检测的代码。
}

catch(异常类  变量){
处理异常的代码;(处理方式)
}

finally{
一定会执行的语句;
}




class Demo{
int div(int a, int b){
return a/b;
}
}

class ExceptionDeamo{
public static void main(String[] args){
demo d = new demo();
try{
int x = d.div(4,0);
System.out.println("x="+x);
}
catch (Exception e)  //Exception e = new ArithmeticException();
{
System.out.println("除零啦");
System.out.println(e.getMessage());          //输出  /by zero
System.out.println(e.toString());             // 异常名称: 异常信息
e.printStackTrace();  //异常名称。异常信息,异常出现的位置。 其实JVM默认的异常处理机制就是在调用printStackTrace方法。打印异常的堆栈信息。

}
System.out.println("over");
}
}




对多异常的处理

1,声明异常时,建议声明更为具体异常,这样处理的可以更具体。

2, 对方声明几个异常,就对应有几个catch块。不要定义多余的catch块。

如果多个catch块中的异常出现继承关系,父类异常catch块放在最下面

先catch小异常,再捕获大异常。

建立在进行catch处理时,catch中一定要定义具体处理方式。

不要简单定义一句e.printStackTrace().

也不要简单的就书写一个打印语句。

class Demo
{
int div(int a, int b)throws ArithmeticException,ArrayIndexOutOfBoundsException
{
return a/b;
}
}

class ExceptionDeamo1
{
public static void main(String[] args)
{
demo d = new demo();
try
{
int x = d.div(4,0);
System.out.println("x="+x);
}
catch (ArithmeticException e)
{
System.out.println("被零除了!!");
System.out.println(e.toString());
}
catch (ArrayIndexOutOfBoundsException e)
{
System.out.println("角标越界了");
System.out.println(e.toString());
}
System.out.println("over");
}
}


编译期异常和运行期异常的区别

Java中的异常被分为两大类:编译时异常和运行时异常。

所有的RuntimeException类及其子类的实例被称为运行时异常,其他的异常就是编译时异常

编译时异常

*Java程序必须显示处理,否则程序就会发生错误,无法通过编译。比如你在处理文件流时的I/O问题,就属于编译时异常。这个时候用thr{}catch 来捕获或者 throws即可。

运行时异常

*程序员自己犯的错误,需要自己回来处理。比如上面提到的ArithmeticException,除数为零,这个属于代码有问题,不处理肯定会报错,如果你操作某个模块发现能正常运行,那只是因为代码还没跑到这个错误的地方而已。。控制台一旦报RuntimeException,就必须要处理。。没有例外的。而且,处理RuntimeException,不是try-catch能解决的。。try-catch在这里使用毫无意义。无需显示处理,也可以和编译时异常一样处理.

编译异常必须在方法上用throws标识,而RunTimeException不用。

throws关键字

定义功能方法时,需要把出现的问题暴露出来让调用者去处理。那么就通过throws在方法上标识。

class Demo
{
int div(int a, int b)throws Exception  // 在功能上通过throws的关键字声明了该功能有可能会出现问题。
{
return a/b;
}
}


throw的概述以及和throws的区别

A:throw的概述

在功能方法内部出现某种情况,程序不能继续运行,需要进行跳转时,就用throw把异常对象抛出。

B:throws和throw的区别

(1)throws

*用在方法声明后面,跟的是异常类名

*可以跟多个异常类名,用逗号隔开

*表示抛出异常,由该方法的调用者来处理

(2)throw

*用在方法体内,跟的是异常对象名

*只能抛出一个异常对象名

*表示抛出异常,由方法体内的语句处理

比如在person类中设置年龄时

public void setAge(int age) throws AgeOutOfBoundsException {
if(age >0 && age <= 150) {
this.age = age;
}else {
//Exception e = new Exception("年龄非法");
//throw e;
throw new Exception("年龄非法");
}
}


finally关键字的特点及作用

A:finally的特点

*被finally控制的语句体一定会执行

特殊情况:在执行到finally之前jvm退出了(比如System.exit(0))

B:finally的作用

*用于释放资源,在IO流操作和数据库操作中会见到

public class Demo7_Finally {

/**
* 除了exit(0)推出JVM虚拟机,否则finally中的代码一定会执行
*return语句相当于是方法的最后一口气,那么在他将死之前会看一看有没有finally帮其完成遗愿,如果有就将finally执行
*后在彻底返回
*/
public static void main(String[] args) {
try {
System.out.println(10/0);
} catch (Exception e) {
System.out.println("除数为零了");
//System.exit(0);                               //退出jvm虚拟机
return;
} finally {
System.out.println("看看我执行了吗");
}
}
}




final,finally和finalize的区别

 1.final

  final修饰类,说明这个类不能被继承,是以个顶级类。

  final修饰变量,说明这个变量是常量。

  final修饰方法,表示这个方法不能被重写,不过可以重载final方法。

 2.finally

  finally是关键字,在异常处理中,try子句中执行需要运行的内容,catch子句用于捕获异常,finally子句表示不管是否发生异常,都会执行。finally可有可无。但是try…catch必须成对出现。

 3.finalize()

  finalize() 是Object类的方法,Java 技术允许使用 finalize() 方法在垃圾收集器将对象从内存中清除出去之前做必要的清理工作。这个方法是由垃圾收集器在确定这个对象没有被引用时对这个对象进行调用。finalize()方法是在垃圾收集器删除对象之前对这个对象调用的子类覆盖 finalize() 方法以整理系统资源或者执行其他清理操作。

如果catch里面有return语句,请问finally的代码还会执行吗?如果会,请问是在return前还是return后。

还会执行,是在return前

public static void main(String[] args) {
Demo d = new Demo();
System.out.println(d.method());
}

}

class Demo {
public int method() {
int x = 10;
try {
x = 20;
System.out.println(1/0);
return x;
} catch (Exception e) {
x = 30;
return x;
} finally {
x = 40;
//return x;                 千万不要在finally里面写返回语句,因为finally的作用是为了释放资源,是肯定会执行的
//如果在这里面写返回语句,那么try和catch的结果都会被改变,所以这么写就是错误的
}
}


最后输出30.因为在执行finally之前, x=30已经被包装进返回路径了,然后执行finally,然后再彻底返回。

自定义异常

因为项目中会出现特有的问题,而这些问题并未被Java所描述并封装对象,不能全抛Exception,所以对于这些特有的问题可以按照java的对问题封装的思想。

将特有的问题,进行自定义的异常封装。

自定义异常

需求:在本程序中,对于除数是-1,也视为是错误的,是无法进行运算的。

那么就需要对这个问题进行自定义描述

当在函数内部出现了throw抛出异常对象,那么就必须要给对应的处理动作。

要么在内部try catch处理

要么在函数上声明让调用者处理

一般情况在,函数内出现异常,函数上需要声明。

class FuShuException extends Exception{
}

class Demo{
int div(int a, int b)throws FuShuException
{
if(b<0)
throw new FuShuException(); //手动通过throw关键字抛出一个自定义异常对象。
return a/b;
}
}

class ExceptionDeamo{
public static void main(String[] args){
demo d = new demo();
try{
int x = d.div(4,0);
System.out.println("x="+x);
}
catch (FuShuException e)  {
System.out.println(e.toString());
System.out.println("除数出现负数了");

}
System.out.println("over");
}
}




发现打印的结果中只有异常的名称,却没有异常的信息,

因为自定义的异常并未定义异常信息。

如何定义异常信息呢?

class FuShuException extends Exception{  //getMessage(); 必须是自定义类继承Exception

private String msg;
FuShuException(String msg) {
this.msg = msg;
}
public String getMessage() {
return msg;
}

}
/*
继承Exception原因:
异常体系有一个特点:因为异常类和异常对象都被抛出。
他们都具备可抛性。这个可抛性是throwable这个体系中独有特点。
只有这个体系汇中的类和对象才可以被throws和throw操作。
*/
class Demo
{
int div(int a, int b)throws FuShuException
{
if(b<0)
throw new FuShuException("出现了除数是负数的情况-----/by fushu"); //手动通过throw关键字抛出一个自定义异常对象。
return a/b;
}
}




因为父类中已经把异常信息的操作都完成了。

所以子类只要在构造时,将异常信息传递给父类通过super语句。

那么就可以直接通过getMessage方法获取自定义的异常信息。

class FuShuException extends Exception  //getMessage();
{
private String msg;
FuShuException(String msg){
super(msg);
}
}


异常的注意事项及如何使用异常处理

A:异常注意事项

*a:子类重写父类方法时,子类的方法必须抛出相同的异常或父类异常的子类。(父亲坏了,儿子不能比父亲更坏)

*b:如果父类抛出了多个异常,子类重写父类时,只能抛出相同的异常或者是他的子集,子类不能抛出父类没有的异常

*c:如果被重写的方法没有异常抛出,那么子类的方法绝对不可以抛出异常,如果子类方法内有异常发生,那么子类只能自己解决,只能try,不能throws

B:如何使用异常处理

(1)原则:如果该功能内部可以将问题处理,用try,如果处理不了,交由调用者处理,这是用throws

(2)区别:

*后续程序需要继续运行就try

*后续程序不需要继续运行就throws

练习

键盘录入一个int类型的整数,对其求二进制表现形式

public class Test2 {

/**
* 键盘录入一个int类型的整数,对其求二进制表现形式
* 如果录入的整数过大,给予提示,录入的整数过大请重新录入一个整数BigInteger
* 如果录入的是小数,给予提示,录入的是小数,请重新录入一个整数
* 如果录入的是其他字符,给予提示,录入的是非法字符,请重新录入一个整数
*
* 分析:
* 1,创建键盘录入对象
* 2,将键盘录入的结果存储在String类型的字符串中,存储int类型中如果有不符合条件的直接报错,无法进行后续判断
* 3,键盘录入的结果转换成int类型的数据,是正确的还是错误的
* 4,正确的直接转换
* 5,错误的要进行对应判断
*/
public static void main(String[] args) {
Scanner sc = new Scanner(System.in);
System.out.println("请输入一个整数:");

while(true) {
String line = sc.nextLine();                    //将键盘录入的结果存储在line中
try {
int num = Integer.parseInt(line);               //将字符串转换为整数
System.out.println(Integer.toBinaryString(num));//将整数转换为二进制
break;                                          //跳出循环
}catch(Exception e) {
try {
new BigInteger(line);
System.out.println("录入错误,您录入的是一个过大整数,请重新输入一个整数:");
}catch (Exception e2) {                         //alt + shif + z (try catch快捷键)
try {
new BigDecimal(line);
System.out.println("录入错误,您录入的是一个小数,请重新输入一个整数:");
} catch (Exception e1) {
System.out.println("录入错误,您录入的是非法字符,请重新输入一个整数:");
}
}

}
}

}

}
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签:  java java se 异常