欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

java笔记(10,java异常)

程序员文章站 2022-07-12 13:27:27
...

 

什么是异常:

异常:就是程序在运行时出现不正常情况。

异常由来:问题也是现实生活中一个具体的事物,也可以通过java的类的形式进行描述,并封装成对象。其实就是java对不正常情况进行描述后的对象体现。

 Throwable

|--Error

|--Exception

        |--RuntimeException

        |--(Other Exception)

Error类层次结构描述了Java运行时系统的内部错误和资源耗尽错误,应用程序不应该抛出这种类型的对象。如果出现了这样的内部错误,除了通告给用户,并尽力使程序安全地终止之外,再也无能为力了。这种情况很少出现。

在设计Java程序事,需要关注Exception层次结构。这个层次结构又分为两个分支;一个分支派生于RuntimeException;另一个分支包含其他异常。划分两个分支的规则是:由程序错误导致的异常属于RuntimeException;而程序本身没有错误,但由于像I/O错误这类问题导致的异常属于其他异常。

如果出现RuntimeExceptin异常,那么就一定是你的问题。

派生于RuntimeException的异常包含下面几种情况:

1,错误的类型转换。

2,数组访问越界。

3,访问空指针。

不是派生于RuntimeException的异常包括:

1,试图在文件尾部后面读取数据。

2,试图打开一个错误格式的URL。

3,试图根据给定的字符串查找Class对象,而这个字符串表示的类根本就不存在。

java语言规范将派生于Error类或RuntimeException类的所有异常称为未检查异常(运行时异常),所有其他的异常称为已检查异常(编译时异常)。

总之,一个方法必须声明所有可能抛出的已检查异常,而未检查异常要么不可控制(Error),要么就应该避免发生(RuntimeException)。如果方法没有声明所有可能发生的已检查异常,编译器就会给出一个错误信息。

通常,应该捕获那些知道如何处理的异常,而将那些不知道怎样处理的异常传递出去。如果想将异常传递出去,就必须在方法的首部添加一个throws说明符,以便告知调用者这个方法可能会抛出异常。

诱饵机制:

当捕获到异常时,可以使用下面这个方法将原始异常包装起来,作为新异常的“诱饵”:

initCause(Throwable e)

当捕获到异常时,可以使用下面的方法重新获得原始异常:

getCause()

如果在一个方法中发生了一个已检查异常,而不允许抛出它,那么包装技术就十分有用,我们可以捕获这个已检查异常,并将它包装成一个运行时异常。

 

异常处理:

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

try{

需要被检测的代码;

}catch{

处理异常的代码;(处理方式)

}finally{

一定会执行的语句;

}

 

finally代码块:定义一定执行的代码;通常用于关闭资源。

记住一点:

catch是用于处理异常的。如果没有catch就代表异常没有处理过,如果该异常是编译时异常,那么就必须申明。

通常,应该捕获那些知道如何处理的异常,而将那些不知道怎样处理的异常传递出去。如果想将异常传递出去,就必须在方法的首部添加一个throws说明符,以便告知调用者这个问题可能会抛出异常。

对捕获到的异常对象进行常见方法操作

s.o.p(e.getMessage());//异常信息

s.o.p(e.toString());//在输出语句里面打印对象,toString可写可不写;  异常名称:异常信息

e.printStackTrace();//异常名称,异常信息:异常出现的位置。

其实jvm默认的异常处理机制,就是在调用printStackTrace方法,打印异常的堆栈的跟踪信息。

 

碰到有问题的方法时,应该要给出预先的处理方式!

 

在函数上声明异常。

便于提高安全性,让调用处进行处理,不处理编译失败!

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

 

对多异常的处理

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

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

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

 

建议:

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

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

也不要简单的就书写一条输出语句。

 

因为项目中会出现特有的问题。

而这些问题并未被java所描述并封装对象。

所以对于这些特有的问题可以按照java的对问题封装的思想。将特有的问题,进行自定义的异常封装。

 

自定义异常:

自定义的异常,我们要手动通过throw关键字抛出一个自定义异常对象。

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

要么在内部try catch处理。

要么在函数上声明让调用者处理。(运用throws关键字。)

 

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

 

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

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

如何定义异常信息呢?

因为父类中已经把异常信息的操作多做完了,

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

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

 

自定义异常必须是自定义类继承Exception。

继承Exception原因:

异常体系有一个特点,异常类和异常对象都需要被抛出。他们都具备可抛性,这个可抛性是Throwable中的独有特点。

只有这个体系中的类和对象才可以被throws和throw操作。

 

throws和throw的区别:

throws使用在函数上;

throw使用在函数内;

 

throws后面跟的异常类,可以跟多个,用逗号隔开。

throw后跟的是异常对象。

 

注意:

Exception中有一个特殊的子类异常RuntimeException 运行异常。

如果在函数内抛出该异常,函数上可以不用声明,编译一样通过。 

如果在函数上声明了该异常,调用者可以不用进行处理,编译一样通过;

之所以不用在函数声明,是因为不需要让调用者处理。当该异常发生,希望程序停止,因为在运行时,出现了无法继续运算的情况,希望停止程序后,对代码进行修正。

 

自定义异常时,如果异常的发生,无法在继续进行运算,就让自定义异常继承RuntimeException。

 

对于异常分为两种:

1,编译时被检测的异常。

2,编译时不被检测的异常(运行时异常,RuntimeException以及其子类。)

 

自定义异常:

定义类继承Exception或者RuntimeException:

1,为了让该自定义类具备可抛性。

2,让该类具备操作异常的共性方法。

 

当要定义自定义异常的信息时,可以使用父类已经定义好的功能。

将异常信息传递给父类的构造函数。

class MyException extends Exception{

MyException(String message){

super(message);

}

}

 

自定义异常,就是按照java的面向对象思想,将程序中出现的特有问题进行封装。

异常的好处:

1,将问题进行封装。

2,将正常流程代码和问题处理代码相分离,方便于阅读。

 

异常的处理原则:

1,处理方式有两种,try或者throws

2,调用到抛出异常的功能时,抛出几个,就处理几个。一个try对应多个catch。

3,多个catch,父类catch放到最下面。

4,catch内,需要定义针对性的处理方式。不要简单的定义printStackTrace,或者输出语句。也不要不写。

当捕获到的异常,本功能处理不了时,可以继续在catch中抛出。

try{

throw new AException();

}catch(AException e){

throw e;

}

 

如果该异常处理不了,但并不属于该功能出现的异常。可以将异常转换后,再抛出和该功能相关的异常。

或者异常可以处理,但需要将异常产生的和本功能相关的问题提供出去,让调用者知道。并处理,也可以将捕获异常处理后,转换新的异常抛出。(举例:汇款失败)

try{

throw new AException();

}catch(AException e){

// 对AException处理。

throw new VException(e);

}

 

异常的注意事项:

在子父类覆盖时:

1,子类抛出的异常必须是父类的异常的子类或者子集。

2,如果父类或者接口没有异常抛出时,子类覆盖出现异常,只能try不能抛。

 

 

异常体系和throw和throws的用法:

异常:

异常是什么?是对问题的描述,将问题进行对象的封装。

 

异常体系:

Throwable

    |--Exception

|--RuntimeException

    |--Error

 

异常体系的特点:异常体系中的所有类以及建立的对象都具备可抛性。也就是说可以被throw和throws关键字所操作。只有异常体系具备这个特点。

 

throw和throws的用法:

throw定义在函数内,用于抛出异常对象。

throws定义在函数上,用于抛出异常类,可以抛出多个并用逗号隔开。

 

当函数内容有throw抛出异常对象,并未进行try处理。必须要在函数上声明,否则编译失败。

注意:RuntimeException除外,也就是说,函数内如果抛出的RuntimeException异常,函数上可以不用声明。

 

如果函数声明了异常,调用者需要进行处理,处理方法可以throws可以try。

 

异常有两种:

编译时被检测异常:

该异常在编译时,如果没有处理(没有抛也没有try),编译失败。

该异常被标识,代表这可以被处理。

 

运行时异常(编译时不检测):

在编译时,不需要处理,编译器不检查。

该异常的发生,建议不处理,让程序停止,需要对代码进行修正。

 

异常处理语句:

try{需要被检测的代码}

catch{处理异常的代码}

finally{一定会执行的代码}

 

有三个结合格式:

1,try{ }catch{ }

2, try{ }finally{ }

3, try{ }catch{ }finally{ }

 

注意:

1,finally中定义的通常是关闭资源代码,因为资源必须释放。

2,finally只有一种情况不会执行,当执行到System.out.exit(0);finally不会执行。

 

java异常处理举例:老师用电脑上课(经典)

 

class LanpinException extends Exception{
  LanpinException(String s){
    super(s);
  }
}

class MaoyanException extends Exception{
  MaoyanException(String s){
    super(s);
  }
}
class KetangException extends Exception{
  KetangException(String s){
  super(s);
  }
}

class Computer{
  private int sta=3;
  void run() throws LanpinException,MaoyanException{
    if(sta==1)
      System.out.println("电脑运行");
    else if(sta==2)
      throw new LanpinException("电脑死机了");
    else if(sta==3)
      throw new MaoyanException("电脑坏了");
  }
  void reset(){
    sta=1;
    System.out.println("电脑重启");
  }
}
class Teacher{
  private String name;
  private Computer cpt;
  Teacher(String name){
    this.name = name;
    cpt = new Computer();
  }
  void spead() throws KetangException{
  try{
    cpt.run();
  }
  catch (LanpinException e){
  cpt.reset();
  }	
  catch (MaoyanException e){
  test();
  throw new KetangException("课程不能继续:"+e.getMessage());
  }
    System.out.println(name+"开始上课");
  }
  void test(){
    System.out.println("做java练习");
  }
}

class ExceptionTest{
  public static void main(String[] args){
  Teacher t = new Teacher("王老师");
   try{
     t.spead();
   }catch (KetangException e){
     System.out.println(e.toString());
     System.out.println("换老师吧");
   }
  }
}