异常

2018-01-07  本文已影响0人  赵小莹

在使用计算机语言进行项目开发的过程中,即使程序员把代码写得尽善尽美,在系统的运行过程中仍然会遇到一些问题,因为很多问题不是靠代码能够避免的,比如:客户输入数据的格式,读取文件是否存在,网络是否始终保持通畅等等。

异常:在Java语言中,将程序执行中发生的不正常情况称为“异常”。(开发过程中的语法错误和逻辑错误不是异常)

Java程序在执行过程中所发生的异常事件可分为两类:Error和Exception
Error和Exception的联系

继承结构:Error和Exception都是继承于Throwable,RuntimeException继承自Exception。

Error和RuntimeException及其子类称为未检查异常(Unchecked exception),其它异常成为受检查异常(Checked Exception)。

Error和Exception的区别

Error类一般是指与虚拟机相关的问题,如系统崩溃,虚拟机错误,内存空间不足,方法调用栈溢出等。如java.lang.StackOverFlowError和Java.lang.OutOfMemoryError。对于这类错误,Java编译器不去检查他们。对于这类错误的导致的应用程序中断,仅靠程序本身无法恢复和预防,遇到这样的错误,建议让程序终止。

Exception类表示程序可以处理的异常,可以捕获且可能恢复。遇到这类异常,应该尽可能处理异常,使程序恢复运行,而不应该随意终止异常。
  1. 空指针访问
  2. 试图读取不存在的文件
  3. 网络连接中断
    对于这些错误,一般有两种解决方法:一是遇到错误就终止程序的运行。另一种方法是由程序员在编写程序时,就考虑到错误的检测、错误消息的提示,以及错误的处理。

捕获错误最理想的是在编译期间,但有的错误只有在运行时才会发生。比如:除数为0,数组下标越界等

Exception又分为运行时异常(Runtime Exception)和受检查的异常(Checked Exception )。
RuntimeException:其特点是Java编译器不去检查它,也就是说,当程序中可能出现这类异常时,即使没有用try……catch捕获,也没有用throws抛出,还是会编译通过,如除数为零的ArithmeticException、错误的类型转换、数组越界访问和试图访问空指针等。处理RuntimeException的原则是:如果出现RuntimeException,那么一定是程序员的错误。
受检查的异常(IOException等):这类异常如果没有try……catch也没有throws抛出,编译是通不过的。这类异常一般是外部错误,例如文件找不到、试图从文件尾后读取数据等,这并不是程序本身的错误,而是在应用环境中出现的外部错误。

分类:编译时异常和运行时异常



1.运行时异常

常见异常:
java.lang.RuntimeException
ClassCastException
ArrayIndexOutOfBoundsException
NullPointerException
ArithmeticException
。。。
java.io.IOExeption
FileNotFoundException
EOFException
java.lang.ClassNotFoundException
java.lang.InterruptedException
java.io.FileNotFoundException
java.sql.SQLException

异常处理机制

在编写程序时,经常要在可能出现错误的地方加上检测的代码,如进行x/y运算时,要检测分母为0,数据为空,输入的不是数据而是字符等。过多的分支会导致程序的代码加长,可读性差。因此采用异常机制。

Java异常处理
Java采用异常处理机制,将异常处理的程序代码集中在一起,与正常的程序代码分开,使得程序简洁,并易于维护。

异常处理是通过try-catch-finally语句实现的。

try{
    ......  //可能产生异常的代码
}
catch( ExceptionName1 e ){
    ......  //当产生ExceptionName1型异常时的处置措施
}
catch( ExceptionName2 e ){
......  //当产生ExceptionName2型异常时的处置措施
}  
[ finally{
......   //无论是否发生异常,都无条件执行的语句
        }  ]
// []代表可以写也可以补血的意思 不出现在代码里

如果明确知道产生的是何种异常,可以用该异常类作为catch的参数;也可以用其父类作为catch的参数。
比如:可以用ArithmeticException类作为参数的地方,就可以用RuntimeException类作为参数,或者用所有异常的父类Exception类作为参数。但不能是与ArithmeticException类无关的异常,如NullPointerException(catch中的语句将不会执行)。

捕获异常的有关信息
与其它对象一样,可以访问一个异常对象的成员变量或调用它的方法。
getMessage() 获取异常信息,返回字符串
printStackTrace() 获取异常类名和异常信息,以及异常出现在程序中的位置。返回值void。
finally
捕获异常的最后一步是通过finally语句为异常处理提供一个统一的出口,使得在控制流转到程序的其它部分以前,能够对程序的状态作统一的管理。

不论在try代码块中是否发生了异常事件,catch语句是否执行,catch语句是否有异常,catch语句中是否有return,finally块中的语句都会被执行.


  public static void main(String[] args) {
        String friends[]={"lisa","bily","kessy"};
        try {
            for(int i=0;i<5;i++) {
                System.out.println(friends[i]);
            }
        }
        catch(ArrayIndexOutOfBoundsException e)  {
            System.out.println("index err");
        }
        System.out.println("\nthis is the end");
  }

输出结果是

lisa
bily
kessy
index err

this is the end
public class DivideZero1{
       int x;
           public static void main(String[] args) {
        int y;
        DivideZero1 c=new DivideZero1();
        try{
            y=3/c.x;
        }
        catch(ArithmeticException e){       
            System.out.println("divide by zero error!");
        }
        System.out.println("program ends ok!");
      }
}

输出结果是

java DivideZero1
divide by zero error!
program ends ok!

不捕获异常时的情况

前面使用的异常都是RuntimeException类或是它的子类,这些类的异常的特点是:
即使没有使用try和catch捕获,Java自己也能捕获,并且编译通过 ( 但运行时会发生异常使得程序运行终止 )。
下面代码,如果不用try-catch处理,循环到第二次就会退出。

public class Main {

    public static void main(String[] args) {
    // write your code here
        Scanner sc = new Scanner(System.in);
        for(int i = 0; i < 5; i++)
        {
            try
            {
                    int k = sc.nextInt();
                    if(i == 1)
                    {
                        k = k / 0;
                    }

                    System.out.println(k);

            }
            catch (Exception ex)
            {
                System.out.println(ex.getMessage());
            }
        }
    }
}

IOException异常处理举例

public class Test6_3{
          public static void main(String[] args)    {
            FileInputStream in=new FileInputStream("myfile.txt");
            int b;
            b = in.read();
            while(b!= -1) {
                    System.out.print((char)b);
                    b = in.read();
            }
          in.close();
         }
}
public class Test6_3{
      public static void main(String[] args){
        try{
            FileInputStream in=new FileInputStream("myfile.txt");
            int b;  
            b = in.read();
            while(b!= -1)       {
                System.out.print((char)b);
                b = in.read();
            }
            in.close();
        }catch (IOException e) {
                System.out.println(e);
        }finally {
                System.out.println(" It’s ok!");
        }
      }
}

编译、运行应用程序Test6_3.java,体会java语言中异常的捕获和处理机制。
相关知识:FileInputStream类的成员方法read()的功能是每次从相应的(本地为ASCII码编码格式)文件中读取一个字节,并转换成0~255之间的int型整数返回,到达文件末尾时则返回-1。

声明抛出异常是Java中处理异常的第二种方式

如果一个方法(中的语句执行时)可能生成某种异常,但是并不能确定如何处理这种异常,则此方法应显示地声明抛出异常,表明该方法将不对这些异常进行处理,而由该方法的调用者负责处理。
在方法声明中用throws语句可以声明抛出异常的列表,throws后面的异常类型可以是方法中产生的异常类型,也可以是它的父类。
声明抛出异常举例:

public void readFile(String file)  throws FileNotFoundException {
        ……
        // 读文件的操作可能产生FileNotFoundException类型的异常
        FileInputStream fis = new FileInputStream(file);
         ..……
     }
public class Test6_4{
    public static void main(String[] args){
        Test6_4 t = new Test6_4();
        try{
            t.readFile();
         }catch(IOException e){   }
    }
    public void readFile() throws IOException {
        FileInputStream in=new FileInputStream("myfile.txt");
        int b;  
        b = in.read();
        while(b!= -1)   {
            System.out.print((char)b);
            b = in.read();
        }
        in.close(); 
    }
}

人工抛出异常

Java异常类对象除在程序执行过程中出现异常时由系统自动生成并抛出,也可根据需要人工创建并抛出。

用户自定义异常类MyException,用于描述数据取值范围错误信息。用户自己的异常类必须继承现有的异常类。

class MyException extends Exception {
    static final long serialVersionUID = 1L;
    private int idnumber;
    public MyException(String message, int id) {
        super(message);
        this.idnumber = id;
    } 
    public int getId() {
        return idnumber;
    }
}
public class Test6_5{
        public void regist(int num) throws MyException {
    if (num < 0) 
              throw new MyException(“人数为负值,不合理”, 3);
    else
              System.out.println("登记人数" + num );
        }
        public void manager() {
    try {
               regist(100);
    } catch (MyException e) {
               System.out.print("登记失败,出错种类"+e.getId());     
    }
    System.out.print("本次登记操作结束");
       }
       public static void main(String args[]){
    Test6_5 t = new Test6_5();
    t.manager();
       }
}
上一篇 下一篇

猜你喜欢

热点阅读