java_异常

时间:2022-02-24 08:35:02

一.什么是异常?

1.当程序”运行后”,当jvm遇到一些无法处理的情况,例如:整数/0,这就表示jvm遇到一种”异常情况”.

通常jvm能够识别这些异常并在控制台打印异常信息,并结束程序

2.为了解决这些情况,java提出了”异常处理机制”,这种机制的使用,可以使得我们的代码在遇到异常时,

可以跳过异常的代码,继续健康的执行下去.

二.异常_异常的产生过程及JVM默认处理异常的方式

1).jvm执行到有异常的代码

2).jvm识别出遇到的异常情况

3).jvm 在类库中寻找描述这种异常的异常类,并创建对象

4).jvm 要看我们的代码中是否希望”捕获”这种异常

是:就执行相应的catch语句

否:在控制台打印异常信息,并结束程序

案例图片:

java_异常

三.异常_Java的异常体系结构及分类

1).Throwable(类):Java中所有异常的顶层类;

       |--Error(错误):不希望程序捕获并处理的错误情况。异常类名都是以:XxxxError结尾的。

       |--Exception(异常):希望程序捕获并处理的异常情况。异常类名都是以:XxxxException结尾的。

           |--RuntimeException(运行时异常):

           |--除RuntimeException以外的其它异常(编译期异常):

--------------------------------------------------------------------------

四.异常的处理_try_catch语句及操作异常对象

1).格式:

        try{

            //可能出现异常的代码

        }catch(异常类名  变量名){

            //如果try中出现了与"异常类名"相同的异常情况,

            //JVM将会执行这个catch,以挽救程序。

3).操作异常对象:

1).getMessage():获取异常信息;

2).toString():异常类名 + 异常信息

3).printStackTrace():打印异常详细信息;【常用:在调试时】

五.第二章:异常的处理_try_catch_catch语句

   1).格式:

        try{

            //可能出现异常的代码

            //多行代码...

        }catch(异常类名1 变量名){

        }catch(异常类名2 变量名){

        }

    2).示例代码: 

public static void main(String[] args) {
int a = 10;
int b = 10;
int[] arr = new int[0];
try {
System.out.println("除法:" + (a / b));//算数运算异常
System.out.println("数组长度:" + arr.length);//空指针异常
System.out.println("数组的第一个元素:" + arr[0]);//数组索引越界异常
} catch (ArithmeticException e) {
System.out.println("算数运算异常!!");
} catch (NullPointerException e) {
System.out.println("空指针异常!!");
} catch (ArrayIndexOutOfBoundsException e) {
System.out.println("数组索引越界异常!!");
} catch (Exception e) {
System.out.println("系统出现了未知异常,请联系系统管理员!");
}
System.out.println("后续代码...");
}

注意:

    1).在try中如果某行出现异常,try中的后续代码将不会被执行;

    2).在多个catch中的异常类名绝对不能重复;

    3).在多个catch中的异常类名通常是同级关系;也可以是子父类关系,如果是子父类关系,父类的异常类名一定要写在多catch的最后。

六.第二章:异常的处理_try_catch_finally语句

1).格式:

try{

//可能出现异常的代码

}catch(异常类名 变量名){

//如果try中出现了与“异常类名”相同的异常,

//将会执行这里

}finally{

                 //无论是否出现异常,都会被执行

                 //通常用在try和catch中有"return"语句的情况。

                 //finally中的代码会保证在try或者catch的return之前被执行。

           }

七.第二章:异常的处理_抛出异常_throw语句

1).格式:

        方法声明(){

            if(....){//判断

                throw new 异常对象("异常信息");//此异常对象,会抛给"调用的代码",由调用的代码处理;

            }

        }

2).示例代码:

  

public class Demo {
public static void main(String[] args) {
try {
getMax(null);
} catch (NullPointerException e) {
System.out.println(e.getMessage());
}
try {
int[] arr = new int[0];
getMax(arr);
} catch (ArrayIndexOutOfBoundsException e) {
System.out.println(e.getMessage());
}
} //编写方法,可以求一个数组中所有元素最大值
public static int getMax(int[] arr) {
if (arr == null) {
//为什么要“抛出一个异常对象”?
//为了告诉调用的代码:我这个方法中出现了异常,没有正常秩序完毕!
throw new NullPointerException("哥们,别给我传空指针啊!揍你啊!!");
}
if (arr.length == 0) {
throw new ArrayIndexOutOfBoundsException("哥们,别给我传0长度数组!削你啊!!");
}
int max = arr[0];
for (int i = 1; i < arr.length; i++) {
max = arr[i] > max ? arr[i] : max;
}
return max;
}
}

八.异常的处理_抛出异常_throws语句

    1).格式:

        方法声明() throws 异常类名1,异常类名2...{

            //正常写代码即可

            //如果方法中出现了声明的异常类名1异常类名2类型的异常,

            //JVM会将这种异常对象抛给调用的代码,由调用的代码处理。

        }

    2).示例代码:

public class Demo {
public static void main(String[] args) {
try {
getMax(null);
} catch (NullPointerException e) {
System.out.println(e.getMessage());
} catch (Exception e) {
e.printStackTrace();
}
try {
int[] arr = new int[0];
getMax(arr);
} catch (ArrayIndexOutOfBoundsException e) {
System.out.println(e.getMessage());
}
} //编写方法,可以求一个数组中所有元素最大值
//throws:表示:声明抛出
public static int getMax(int[] arr) throws NullPointerException,
ArrayIndexOutOfBoundsException{
//方法中正常写代码即可。
int max = arr[0];
for (int i = 1; i < arr.length; i++) {
max = arr[i] > max ? arr[i] : max;
}
return max;
}
}

3).注意事项

        1).如果throws声明抛出的是运行时异常(RuntimeException及其子类)调用处可以不处理编译通过。

           但运行时,如果真的出现异常,仍然停掉程序。

        2).如果throws声明抛出的是:编译期异常(除RuntimeException以外的异常),调用处必须处理,

           要么try...catch...,要么继续抛出,否则编译错误。

throw和throws的区别:

    1).throw:(抛出)

1).用在方法中;

2).throw后面是:new 异常对象();而且只能有一个;

3).表示:立即抛出一个异常对象给方法的调用处;

    2).throws:(声明抛出)【常用】

        1).用在方法的声明处;

2).throws后面是:“异常类名”,可以有多个。

3).表示:告诉JVM,此方法内部“可能会出现某种异常”,如果运行时,真的出现异常,JVM

要将这个异常对象抛给调用处。

九.异常的处理_注意事项

    1).子类重写父类方法时,throws的写法:

class Fu{

public void show1(){}

public void show2() throws NullPointerException{}//空指针异常--运行时异常

public void show3() throws IOException{}//编译期异常

}

class Zi extends Fu{

public void show1() throws Exception{}

public void show2() throws ArrayIndexOutOfBoundsException{}

public void show3() throws IOException,ParseException{}

}

    2).当子类重写父类方法时,throws方面总体遵循以下三条规则:

        1).无论父类方法是否抛出异常,子类方法都可以:不抛出任何异常;

        2).无论父类方法是否抛出异常,子类方法都可以:抛出任何的运行时异常

        3).无论父类方法是否抛出异常,子类方法可以抛出跟父类相同的编译期异常(或者子类异常),

           不能抛出比父类更多的编译期异常。

3).示例代码:

class Fu{
public void show1() {}
public void show2() throws NullPointerException{}//空指针异常--运行时异常
public void show3() throws IOException {}//编译期异常
}
//1.都可以不抛出任何异常;
class Zi1 extends Fu{
@Override
public void show1() {
super.show1();
}
@Override
public void show2() {
super.show2();
}
@Override
public void show3() {
}
}
//2.都可以抛出任何的运行时异常
class Zi2 extends Fu{
@Override
public void show1() throws NullPointerException { }
@Override
public void show2() throws NullPointerException { }
@Override
public void show3() throws NullPointerException {
super.show1();
}
}
//3.只能抛出跟父类相同的“编译期异常(或者子类)”,不能抛出父类没有的编译器异常。
class Zi3 extends Fu{
@Override
public void show1() {
}
@Override
public void show2() {
}
@Override
public void show3() throws ChangedCharSetException,ActivationException {//错误
}
}

十.异常_自定义异常

    1).有些时候,根据自身的业务逻辑,需要某种异常类,但类库中没有提供这种异常类,这时我们可以自定义自己的异常类。

            class Student{

                private int age;

                public void setAge(int age){

                    if(age < 15 || age > 50){

                        //抛出异常

                        throw new 自定义异常对象();

                    }

                    this.age = age;

                }

            }

    2).定义异常类

        class Throwable{

            private String detailMessage;

            public String getMessage(){

                this.detailMessage;

            }

        }

        class Exception extends Throwable{

            public Excepton(String msg){

                super(msg);

            }

        }

        class RuntimeException extends Exception{

            public RuntimeException (String msg){

                super(msg);

            }

        }

        //自定义异常类可以继承自任意的Exception类及其子类。

        class AgeException extends RuntimeException{

            public AgeException (String msg){

                super(msg);

            }

        }

public class Student{
private int age;
public void setAge(int age){
if(age < 15 || age > 50){
//抛出异常
throw new AgeException("年龄非法,必须在15到50之间!!");
}
this.age = age;
}
}
public class Demo {
public static void main(String[] args) {
Student stu = new Student();
try {
stu.setAge(10);
} catch (NullPointerException e) {
System.out.println("异常了:");
System.out.println(e.getMessage());
}
}
}