java笔记(10,java异常)

阅读更多

 

什么是异常:

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

异常由来:问题也是现实生活中一个具体的事物,也可以通过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("换老师吧");
   }
  }
}

 

 

 

 

 

 

你可能感兴趣的:(java,exception,异常)