黑马程序员-java基础(三)-异常体系

------- android培训java培训、期待与您交流! ----------

异常体系

异常的体系
异常:是在运行时期发生的不正常情况,java通过面向对象的思想将问题封装成了对象,用异常类对其进行描述。
Throwable:Throwable 类是 Java 语言中所有错误或异常的超类
特点:Throwable及其所有的子类都具有可抛性
不正常情况分成了两大类:
1. 一般不可处理的:Error
是由jvm抛出的严重性问题。 这种问题发生,一般不针对性处理,直接修改程序。
2. 可以处理的:Exception
Throwable中的方法:
    1. getMessage():获取异常信息,返回字符串。
    2. toString():获取异常类名和异常信息,返回字符串。
    3. printStackTrace():获取并打印异常类名和异常信息,以及异常出现在程序中的位置,返回值void。
    4. printStackTrace(PrintStream s):通常用该方法将异常内容保存在日志文件中,以便查阅。

    示例:
class Demo{
       public static int method(int[] arr, int index){
             if(arr == null){
                   throw new NullPointerException("数组的引用不能为空!");
             }
             if(index >= arr.length ){
                   throw new ArrayIndexOutOfBoundsException("数组的角标越界:" + index);
             }
             return arr[index];
       }
}
class Test{
       public static void main(String[] args){
            int[] arr = new int[3];
            Demo.method(arr,30);
       }
}
 
 
运行结果:

异常处理

java提供特有的语句进行处理 try { 需要被检测 的代码 } catch(异常类 变量) { 处理 方式 }
finally{一定会执行语句}
P.S.处理过程:    try中检测到异常会将异常对象传递给catch,catch捕获到异常进行处理。    finally里通常用来关闭资源。比如:数据库资源,IO资源等。    需要注意:try是一个独立的代码块,在其中定义的变量只在该变量块中有效。    如果在try以外继续使用,需要在try外建立引用,在try中对其进行初始化。IO,Socket就会遇到。P.S.    finally代码块只有一种情况不会被执行,就是在之前执行了System.exit(0)。
class ExceptionTest{
    int method(int a, int b)
    {
        return a/b; 
    }
}
class Test{
    public static void main(String[] args)
    {
        ExceptionTest et=new ExceptionTest();
        try //扑捉异常
        {
            int x=et.method(4,0);
            System.out.println("x="+x);
        } 
        catch (Exception e) //处理异常
        {
            System.out.println("除零啦");
            System.out.println(e.getMessage());
            System.out.println(e.toString());
            //e.printStackTrace();
        }
        System.out.println("over");
    }
}
 
  
 
 
 
运行结果:

自定义异常

自定义异常:    可以自定义出的问题称为

P.S.

如果让一个类成为异常类,必须要继承异常体系,因为只有成为异常体系的子类才有资格具备可抛性,才可以被两个关键字所操作:throws、throw。

【例】对于角标为负数的情况,可以用负数角标异常来表示,负数角标这种异常在java中并没有定义过。按照java异常的创建思想,面向对象,将负数角标进行自定义描述,并封装成对象

 

class FushuIndexException extends Exception//自定义异常类继承异常体系
	{
	FushuIndexException(String message)
	{
		super(message);//父类已经将操作异常信息完成thread(string s)
	}
	}

class ExceptionTest 
{
	public int method(int a,int b)throws FushuIndexException
	{
		if(b<0)
		throw new FushuIndexException("出现负数啦");//抛出异常
		return(a/b);
	}
}

class Test
{
    public static void main(String[] args){
    	ExceptionTest et=new ExceptionTest();
    	try {
    		int x=et.method(4,-1);
        	System.out.println(x);
		} catch (Exception e) 
    	{
			System.out.println(e.toString());
		}    	
    }
}

运行结果:


         throws和throw的区别:

1. throws用于标识函数暴露出的异常类,并且可以抛出多个,用逗号分隔。throw用于抛出异常对象。
2. thorws用在函数上,后面跟异常类名。throw用在函数内,后面跟异常对象。

总结

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

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


异常的分类】

1. 编译时被检测异常:只要是Exception和其子类都是,除了特殊子类RuntimeException体系。这种问题一旦出现,希望在编译时就进行检测,让这种问题有对应的处理方式。这样的问题都可以针对性的处理。

2. 编译时不检测异常(运行时异常):就是Exception中的RuntimeException和其子类。这种问题的发生,无法让功能继续,运算无法运行,更多是因为调用的原因导致的或者引发了内部状态的改变导致的。那么这种问题一般不处理,直接编译通过,在运行时,让调用者调用时的程序强制停止,让调用者对代码进行调整。

所以自定义异常时,要么继承Exception,要么继承RuntimeException。

    示例:

    class FuShuIndexException extends RuntimeException{
          FuShuIndexException(){}

          FuShuIndexException(String msg){
                 super(msg);
          }
    }

    class Demo{
           public static int method(int[] arr, int index){//RuntimeException没有必要用throws抛出,并不是必须要处理
                 if(index < 0){
                       throw new FuShuIndexException("数组的角标是负数啦!" );
                 }
                 return arr[index];
           }
    }

   运行结果:


  P.S.
    RuntimeException是那些可能在Java虚拟机正常运行期间抛出的异常的超类。
    可能在执行方法期间抛出但未被捕获的RuntimeException的任何子类都无需在throws子句中进行声明。

【例】

class FuShuIndexException extends Exception{
	FuShuIndexException(String msg){
           super(msg);
    }
}

class ExceptionDemo{
     public static int method(int a, int b)throws NullPointerException,FuShuIndexException{
           if(b==0)
                 throw new NullPointerException("除数没有值");
           if(b < 0){
                 throw new FuShuIndexException("除数为负");
          }
           return a/b;
    }
}

class Test{
     public static void main(String[] args){
          try{
                int num = ExceptionDemo.method(4,-1);
                System.out.println("num:" + num);
         } catch(NullPointerException e){
                System.out.println(e);
          } 
          catch(FuShuIndexException e){
                System. out.println("message:" + e.getMessage());
                System.out.println("string:" + e);
                e.printStackTrace(); //jvm 默认的异常处理机制就是调用异常对象的这个方法。
                System.out.println("被除数为负!!!");
          } catch(Exception e){//Exception的catch放在最下面,先处理有针对性的异常
                System.out.println(e);
          }
          System.out.println("over" );
    }
}

运行结果:


  异常处理的原则:
1. 函数内容如果抛出需要检测的异常,那么函数上必须要声明。否则,必须在函数内用try/catch捕捉,否则编译失败。
2. 如果调用到了声明异常的函数,要么try/catch,要么throws,否则编译失败。
3. 什么时候catch,什么时候throws呢?
功能内容可以解决,用catch。
解决不了,用throws告诉调用者,由调用者解决。

 4. 一个功能如果抛出了多个异常,那么调用时,必须有对应多个catch进行针对性处理。
内部有几个需要检测的异常,就抛几个异常,抛出几个,就catch几个。


    示例:
class ExceptionDemo{
       public int show(int index) throws ArrayIndexOutOfBoundsException{
             if(index < 0)
                   throw new ArrayIndexOutOfBoundsException("越界啦!");
             int[] arr = new int[3];
             return arr[index];
       }
}

class Test{
       public static void main(String[] args){
    	   ExceptionDemo d = new ExceptionDemo();
            try{
                  int num = d.show(-3);
                  System.out.println("num = " + num);
             } catch(ArrayIndexOutOfBoundsException e){
                  System.out.println(e.toString());
                  System.exit(0);//退出jvm
             } finally{//通常用于关闭(释放)资源
                  System.out.println("finally");//由于前面执行了System.exit(0);,故不会执行此语句。
             }
            System.out.println("over");
      }
}

<span style="font-size:14px;">P.S.</span>

try catch finally 代码块组合特点:

1. try catch finally
2. try catch(多个):当没有资源需要释放时,可以不用定义finally。

3. try finally:异常无法直接catch处理,但是资源必须关闭。

【例】

/*
 毕老师用电脑上课。

 问题领域中涉及两个对象。
 毕老师,电脑。

 分析其中的问题。
 比如电脑蓝屏,冒烟等。
 */
 class LanPingException extends Exception{
       LanPingException(String msg){
              super(msg);
       }
 }

 class MaoYanException extends Exception{
       MaoYanException(String msg){
              super(msg);
       }
 }

 class NoPlanException extends Exception{
       NoPlanException(String msg){
              super(msg);
       }
 }

 class Computer{
        private int state =1;//0 1  2
        public void run() throws LanPingException,MaoYanException{
              if(state == 1)
                    throw new LanPingException("电脑蓝屏啦!");
              if(state == 2)
                    throw new MaoYanException("电脑冒烟啦!");
              System. out.println("电脑运行");
        }
        public void reset(){
             state = 0;
             System.out.println("电脑重启");
        }
 }

 class Teacher{
        private String name ;
        private Computer comp ;

        Teacher(String name){
              this.name = name;
              comp = new Computer();
        }

        public void prelect() throws NoPlanException{
              try{
                   comp.run();
                   System. out.println(name + "讲课");
              } catch(LanPingException e){
                   System.out.println(e.toString());
                   comp.reset();
                   prelect();
              } catch(MaoYanException e){
                   System. out.println(e.toString());
                   test();
                    //可以对电脑进行维修
                    throw new NoPlanException("课时进度无法完成,原因:" + e.getMessage());
              }
        }
        public void test(){
             System.out.println("大家练习");
        }
 }

 class Test{
        public static void main(String[] args){
             Teacher t = new Teacher("毕老师");
              try{
                   t.prelect();
              } catch(NoPlanException e){
                   System.out.println(e.toString() + "......." );
                   System.out.println("换人");
              }
        }
 }

state=1运行结果:


state=2运行结果:


异常的注意事项:   

    1. RuntimeException以及其子类如果在函数中被throw抛出,可以不用在函数上声明。
    2. 子类在覆盖父类方法时,父类的方法如果抛出了异常,那么子类的方法只能抛出父类的异常或者该异常的子类。
    3. 如果父类抛出多个异常,那么子类只能抛出父类异常的子集。
    简单说:子类覆盖父类只能抛出父类的异常或者子类的子集。

    P.S.

    如果父类的方法没有抛出异常,那么子类覆盖时绝对不能抛,就只能try。





  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值