重点
1.异常相关概述
1.1 什么是异常
异常就是Java程序在运行过程中出现的错误。
前面接触过的空指针,数组越界,类型转换错误异常等
1.2 Throwable
Throwable 类是 Java 语言中所有错误或异常的超类。
只有当对象是此类(或其子类之一)的实例时,才能通过 JVM 或者 throw 语句抛出。
1.3 异常的继承体系
1.4 JVM默认是如何处理异常的?
- jvm有一个默认的异常处理机制,就将该异常的名称、异常的信息、异常出现的位置打印在了控制台上,同时程序停止运行。
1.5 Java处理异常的两种方式
Java虚拟机处理
自己处理
1.6 为什么会有异常
eg:因为你不知道未来会怎么样,需要做个准备(写代码也是,你需要对一些未知的东西做下处理),比如你去旅游,不知道会下雨还是出大太阳,你需要准备雨伞和帽子.如果下雨,我有雨伞可用,如果出大太阳,可以用帽子,也有可能天气刚刚好,雨伞和帽子都用不着,这叫白带。
1.7 回顾几个常见异常
2. try-catch
自己处理异常的两种方试
try…catch…finally
throws
2.1 try-catch异常处理方式
2.2 try-catch-catch 多个catch处理方式
try {
int[] arr = {1,2,3};
System.out.println(arr[4]);//ArrayIndexOutOfBoundsException数组越界异常
int a = 10 / 0;//ArithmeticException:算术异常
System.out.println(a);
int[] arr1 = null;
System.out.println(arr1[0]);//NullPointerException空指针
} catch (ArrayIndexOutOfBoundsException e) {
System.out.println("数组越界异常...");
} catch (ArithmeticException e){
System.out.println("算术异常...");
} catch(NullPointerException e){
System.out.println("空指针异常...");
}
2.3 try-catch-catch 多个异常-另一种写法
把多个异常放在一个catch中,用|或符号连接
try {
int[] arr = {1,2,3};
System.out.println(arr[4]);//ArrayIndexOutOfBoundsException数组越界异常
int a = 10 / 0;//ArithmeticException:算术异常
System.out.println(a);
//catch (ArrayIndexOutOfBoundsException | ArithmeticException | NullPointerException e)
} catch (ArrayIndexOutOfBoundsException | ArithmeticException e) {
System.out.println(e.getClass());
System.out.println("数组越界异常或者算法异常...");
}
2.4 多个异常处理的一个注意事项
catch中父类的Exception只能放在最后面
catch一个原则,先由子类处理异常,子类不能处理异常,再由父类处理
try {
//1.有可能出现数组越界异常
int[] arr = {1,2,3};
System.out.println(arr[1]);
//2.算术异常
int a = 10 / 2;
//3.空指针异常
int[] arr1 = null;
//NullPointerException np;
System.out.println(arr1[0]);
}catch (ArrayIndexOutOfBoundsException e) {
System.out.println("数组越界异常");
} catch (ArithmeticException e) {
System.out.println("算术异常");
} catch (Exception e) {
System.out.println("其它异常");
}
2.5"编译时异常"和"运行时异常"
Java中的异常被分为两大类:“编译时异常"和"运行时异常”。
编译时异常和运行时异常的区别
>所有的RuntimeException类及其子类被称为运行时异常
>其他的异常就是编译时异常
编译时异常
Java程序必须显示处理,否则程序就会发生错误,无法通过编译
运行时异常
无需显示处理,也可以像编译时异常一样处理
比如算术异常,数组越界,空指针都是运行时异常
2.6 Throwable有几个常用的方法
Exception是一个继承Throwable的子类
getMessage():获取异常信息,返回字符串。
toString():获取异常类名和异常信息,返回字符串。
printStackTrace():打印异常类名和异常信息,以及异常出现在程序中的位置
3.throws
3.1 throws方式处理异常
实现方式:
- 定义方法时,在方法名后面添加throws 异常类名,相当于问题暴露出来让调用者去处理
- 方法内部要写个throw 异常对象,来抛出一个异常对象
Person.java
class Person {
private int age;
public void setAge(int age)throws Exception {
//年龄要1~150岁内
if(age >= 1 && age <=150){
this.age = age;
}else{
//System.out.println("你是来自火星");
throw new Exception("你是来自火星");//告诉外界异常的类型
}
}
public void say(){
System.out.println("我今年"+ age);
}
}
main方法
3.2 throws和throw的区别
throws
用在方法声明后面,跟的是异常类名
可以跟多个异常类名,用逗号隔开
它表示抛出异常,由该方法的调用者来处理
throw
用在方法体内,跟的是异常对象名
只能抛出一个异常对象名,表示抛出异常
4.finally
4.1 try-catch-finally
finally的特点
finally的语句体一般情况下一定会执行
特殊情况:在执行到finally之前jvm退出了(比如System.exit(0))
finally的作用
用于释放资源,在IO流操作和数据库操作中会见到
5. 工作中如何使用异常处理
1.原则:如果自己能处理的问题,就用try-catch, 如果自己不能解决的问题,就throws
2.try-catch和throws的区别:
后续程序需要继续运行就用try
后续程序不需要继续运行就throws
3.如果JDK没有提供对应的异常,需要自定义异常。
练习题
1. 自定义年龄异常
/**
* 自定义异常的步骤:
* 1.写一个类(AgeException)继承Exception,这种是编译时异常
* 2.在子类异常AgeException提供一个带字符串参数的构造方法
* public AgeException(String msg){
super(msg);
}
3.自定义异常时,也可以继承RuntimeException,这种是运行时异常
* @author gyf
*
*/
public class Demo01 {
public static void main(String[] args){
//自定义异常概述和基本使用
/**
* 1.自定义一个年龄异常AgeException
*/
//ArithmeticException ae;
//int a = 10 / 0;//抛出Exception的子类
Person p = new Person();
p.setAge(180);
p.say();
}
}
class Person{
private int age;
/**
* age在1~150岁范围
* 在方法里抛出异常
* @param age
*/
public void setAge(int age) throws AgeException{
if(age >=1 && age <= 150){
this.age = age;
}else{
throw new AgeException("你是火星来的,年龄不合法");//抛出异常对象
}
}
public void say(){
System.out.println("今年" + age + "岁");
}
}
//编译时异常
/*class AgeException extends Exception{
public AgeException(String msg){
super(msg);
}
}*/
//运行时异常
class AgeException extends RuntimeException{
public AgeException(String msg){
super(msg);
}
}
2. 自定义异常注意事项
public class Demo01 {
public static void main(String[] args) {
/*异常注意事项
1.子类重写父类方法时,子类的方法必须抛出相同的父类异常
2.如果被重写的方法没有异常抛出,那么子类的方法最好不要抛出异常
3.如果子类方法内有异常发生,那么子类只能try,不能throws*/
}
}
class A{
public void test(){
}
}
class B extends A{
@Override
public void test() {
// TODO Auto-generated method stub
try {
FileInputStream fis = new FileInputStream("a.txt");
} catch (FileNotFoundException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
//年龄异常
class AgeException extends Exception{
}
3.自定义一个登录异常
注意事项:什么时候方法后面不需要写throws?
方法内throw的异常对象如果是一个运行时异常,方法后面就不会写throws
/**
* 什么时候方法后面不需要写throws
* 方法内throw的异常对象如果是一个运行时异常,方法后面就不会写throws
* @author gyf
*
*/
public class Demo01 {
public static void main(String[] args) {
// TODO Auto-generated method stub
//案例:自定义一个登录异常
/**
* 1、当用户名和密码不正确时抛出登录异常
* 2、用户名admin,密码为123
* 3、用户名和密码由用户通过键盘输入
*/
//1.获取用户的输入的用户名和密码
Scanner scanner = new Scanner(System.in);
System.out.println("请输入用户名:");
String username = scanner.nextLine();
System.out.println("请输入密码:");
String password = scanner.nextLine();
//2.判断用户名和密码是否正确
if("admin".equals(username) && "123".equals(password)){
System.out.println("登录成功");
}else{
//3.抛出登录异常
throw new LoginException("用户名或者密码不正确");
}
}
}
public class LoginException extends RuntimeException{
public LoginException(String message) {
super(message);
// TODO Auto-generated constructor stub
}
}
4. 键盘输入的用户名只能包含英文字符
public class Demo01 {
public static void main(String[] args) {
/*练习:键盘输入的用户名只能包含英文字符
1.使用键盘获取用户录入内容,只能传入英文字符串,否则抛出 非法字符 的异常
2.非法字符 的异常需要自定义
3.异常定义为运行时异常*/
//1.获取用户的输入
Scanner scanner = new Scanner(System.in);
System.out.println("请输入用户名");
String username = scanner.nextLine();
//2.判断字符是否合法
for(int i=0;i<username.length();i++){
char ch = username.charAt(i);
//小写字符
boolean isSmall = (ch >= 'a' && ch <= 'z');
//大写字符的区间
boolean isBig = (ch >= 'A' && ch <= 'Z');
//第一种写法
/*if(isSmall || isBig){
//System.out.println(ch + "合法字符");
}else{
//System.out.println(ch + "不合法字符");
throw new IllegalCharacterException("不合法的字符:" + ch);
}*/
if(!(isSmall || isBig)){//不合法字符
throw new IllegalCharacterException("不合法的字符:" + ch);
}
}
}
}
//定义 非法字符 的异常
class IllegalCharacterException extends RuntimeException{
public IllegalCharacterException(String message) {
super(message);//RuntimeException(String message)
}
}
面试题
1. final,finally和finalize的区别
final可以修饰类,不能被继承;修饰方法,不能被重写;修饰变量,只能赋值一次
finally是try语句中的一个语句体,不能单独使用,用来释放资源
finalize是一个方法,当垃圾回收器确定不存在对该对象的更多引用时,由对象的垃圾回收器调用此方法。【当对象从内存中消失会调用这个方法】
2. catch中关于return的面试题
如果catch里面有return语句,请问finally的代码还会执行吗?如果会,请问是在return前还是return后?
答:会执行,finally的代码在return之前执行
总结
通过今天的学习,对Java 的异常类有了更深的了解,处理异常有两种:JVM和自己处理。自己处理的有两种:try catch finally 和 throws两种,前者可以运行后面的代码,后者不能运行语句后的代码。异常又分运行时异常和编译时异常,前者再运行时出现,如RuntimeException,后者是编译的时候出现,就是写代码就报错。在自定义异常时会根据情况继承运行时异常和编译时异常。然后通过自定义一个登录异常、 键盘输入的用户名只能包含英文字符两个案例,对异常可以灵活运用。