黑马程序员_面向对象(3) 异常处理机制

本文详细阐述了Java中异常的概念、类型、处理方法及应用,包括Error与Exception的区别、异常处理语句try-catch-finally的应用、自定义异常、异常在子父类覆盖中的体现及异常在实际开发中的应用案例。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

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

一.异常概述

   异常:就是程序在运行时出现不正常情况。
   异常由来:问题也是现实生活中一个具体的事物,也可以通过java的类的形式进行描述。并封装成对象。
   其实就是java对不正常情况进行描述后的对象体现。
   对于问题的划分:两种:一种是严重的问题,一种非严重的问题。
   对于严重的,java通过Error类进行描述。
   对于Error一般不编写针对性的代码对其进行处理。
   对与非严重的,java通过Exception类进行描述。
   对于Exception可以使用针对性的处理方式进行处理。
   无论Error或者Exception都具有一些共性内容。
   比如:不正常情况的信息,引发原因等。
   Throwable
   |--Error
   |--Exception
       |---RuntimeException
 面试题: Error 和 Exception 区别?
 
Error表示系统级的错误和程序不必处理的异常, 

 Exception表示需要捕捉或者需要程序进行处理的异常。

异常有两种:

编译时被检测异常

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

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

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

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

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

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

二.异常处理
java 提供了特有的语句进行处理。
try
{
需要被检测的代码;
}
catch(异常类 变量)
{
处理异常的代码;(处理方式)
}
finally
{
一定会执行的语句;
}
 

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

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


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

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

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

多个异常处理
1.声明异常时,建议声明更为具体的异常。这样处理的可以更具体。
2.对方声明几个异常,就对应有几个catch块。不要定义多余的catch块。
   如果多个catch块中的异常出现继承关系,父类异常catch块放在最下面。

建立在进行catch处理时,catch中一定要定义具体处理方式。
不要简单定义一句 e.printStackTrace(),
也不要简单的就书写一条输出语句。

对捕获到的异常对象进行常见方法操作。
String getMessage():获取异常信息。

在函数上声明异常。
便于提高安全性,让调用出进行处理。不处理编译失败

注意:

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

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


自定义异常方法。

定义类继承Exception或者RuntimeException

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

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

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

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

eg:

	class MyException extends Exception
	{
		MyException(String message)
		{
			super(message);
		}
	}
自定义异常信息

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

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

eg:

class FuShuException extends Exception //getMessage();
{
	private int value;

	FuShuException()
	{
		super();
	}
	FuShuException(String msg,int value) // 自定义异常信息
	{
		super(msg);
		this.value = value;
	}

	public int getValue()
	{
		return value;
	}

}
class Demo
{
	int div(int a,int b)throws FuShuException
	{
		if(b<0) // 手动抛出异常
			throw new FuShuException("出现了除数是负数的情况------ / by fushu",b);//手动通过throw关键字抛出一个自定义异常对象。

		return a/b;
	}
}


异常在子父类覆盖中的体现;

1,子类在覆盖父类时,如果父类的方法抛出异常,那么子类的覆盖方法,只能抛出父类的异常或者该异常的子类。

2,如果父类方法抛出多个异常,那么子类在覆盖该方法时,只能抛出父类异常的子集。

3,如果父类或者接口的方法中没有异常抛出,那么子类在覆盖方法时,也不可以抛出异常。

如果子类方法发生了异常。就必须要进行try处理。绝对不能抛


异常应用:

// 面向对象思想封装自定义异常为对象
class LanPingException extends Exception
{
	LanPingException(String message)
	{
		super(message);
	}
}
// 自定义异常
class MaoYanException extends Exception
{
	MaoYanException(String message)
	{
		super(message);
	}
}
class NoPlanException extends Exception
{
	NoPlanException(String msg)
	{
		super(msg);
	}
}

class Computer
{
	private int state = 3;
	public void run()throws LanPingException,MaoYanException
	{
		if(state==2)
                      // 抛出检测异常,方法必须声明异常
			throw new LanPingException("蓝屏了");
		if(state==3)
			throw new MaoYanException("冒烟了");
		System.out.println("电脑运行");
	}
	public void reset()
	{
		state = 1;
		System.out.println("电脑重启");
		
	}
}

class Teacher
{
	private String name;
	private Computer cmpt;

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

	}

	public void prelect()throws NoPlanException
	{
		try
		{
			cmpt.run();			
		}
		catch (LanPingException e)
		{
			cmpt.reset();
		}
		catch (MaoYanException e)
		{			
			test();
			throw new NoPlanException("课时无法继续"+e.getMessage());			
		}
		System.out.println("讲课");
	}
	public void test()
	{
		System.out.println("练习");
	}

}
class ExceptionTest 
{
	public static void main(String[] args) 
	{
		Teacher t = new Teacher("毕老师");
		try
		{
			t.prelect();
		}
		catch (NoPlanException e)
		{
			System.out.println(e.toString());
			System.out.println("换老师或者放假");
		}
		
	}
}
-------------------------------------------------------------------------------------------------------------------------

三 .面试题
  黑马论坛里看到一道面试题:
 
private static void foo() {

try
{
        System.out.println("try");
    foo();
}catch (Throwable e) {
    System.out.println("catch");
    foo();
}finally {
    System.out.println("finally");
    foo();
}
}
public static void main(String[] args) {
    foo();
}
 据java内存模型分析如下:
 
 
   当main 调用 foo() ,foot() 再次调用foo(),一直填满第n层的栈帧,即将在try中再次调用foo()方法,并且
   JVM 为此分配内存, 即n+1层栈帧, 此时,catch 抛出异常 StackOverFlowError,此时又被catch捕获处理,
   返回之前调用n层出的finally语句,再finally 里又次调用foo() , JVM又要  抛出  StackOverFlowError,在第n层
   foo()方法抛出异常并不代表结束点,catch为其延长生命周期, 又回到第n+1层,一直循环下去.... 运行结果
   在不同JDK,不同pc配置是不同了,个人pc 运行结果   try  try try.....finally finally... 

-------android培训java培训、期待与您交流! ----------
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值