异常指在方法不能按正常方式完成时,可以通过抛出异常的方式退出该方法。在异常中封装了方法执行过程中的错误信息及原因,调用方在获取该异常后可根据业务的情况选择处理该异常或继续抛出该异常。
在方法执行中出现异常时,Java 异常处理机制会将代码的执行权交给异常处理器,异常处理器根据在系统中定义的异常处理规则执行不同的异常处理逻辑(抛出异常或捕捉并处理异常)。
在 Java 中,Throwable 是所有错误或异常的父类,Throwable 又可分为 Error 和 Exception,常见的 Error 有 AWTError、ThreadDeath,Exception 又可分为 RuntimeException(运行时异常)和 CheckedException(检查异常),如下:
Error 指 Java 程序运行错误。如果程序在启动时出现 Error,则启动失败;如果程序在运行过程中出现 Error,则系统将退出进程。出现 Error 通常是因为系统的内部错误或资源耗尽,Error 不能在运行过程中被动态处理。如果程序出现 Error,则系统能做的工作也只能是记录错误的成因和安全终止。
Exception 指 Java 程序运行异常,即运行中的程序发生了人们不期望发生的事件,可以被 Java 异常处理机制处理。Exception 也是程序开发中异常处理的核心,如图:
处理异常有抛出异常和使用 try catch 语句块捕获并处理异常这两种方式。
(1)抛出异常:指遇到异常时不进行具体处理,而是将异常抛给调用者,由调用者根据情况处理。有可能是直接捕获并处理,也有可能是继续向上层抛出异常。抛出异常有三种方式:throws、throw、系统自动抛出异常。其中,throws 作用在方法上,用于定义方法可能抛出的异常;throw 作用在方法内,表示明确抛出一个异常。
throw 和 throws 的具体区别如下:
(2)使用 try catch 语句块捕获并处理异常:使用 try catch 语句块捕获异常能够有针对地处理每种可能出现的异常,并在捕捉到异常后根据不同的情况做不同的处理。
相关面试题:
Java 的反射机制可以动态获取类和对象的信息,以及动态调用对象的方法,被广泛应用于动态代理的场景中。
动态语言指程序在运行时可以改变其结构的语言,比如新的属性或方法的添加、删除等结构上的变化。JavaScript 、Ruby 、 Python 等都属于动态语言;C、C++ 不属于动态语言。从反射的角度来说,Java 属于半动态语言。
反射机制指在程序运行过程中,对任意一个类都能获取其所有属性和方法,并且对任意对象都能调用其任意方法。这种动态获取类和对象的信息,以及动态调用对象的方法的功能被称为 Java 的反射机制。
Java 中的对象有两种类型:编译时类型和运行时类型。编译时类型指在声明对象时采用的类型,运行时类型指为对象赋值时所采用的的类型。
在如下代码中,persion 对象的编译时类型为 Persion ,运行时类型为 Student ,因此无法在编译时获取在 Student 类中定义的方法:
Persion persion = new Student();
因此,程序在编译期间无法预知该对象和类的真实信息,只能通过运行时信息来发现该对象和类的真实信息,而其真实信息(对象的属性和方法)通常通过反射机制来获取,这便是 Java 中反射机制的核心功能。
Java 的反射 API 主要用于在运行过程中动态生成类、接口或对象等信息,其常用 API 如下:
反射的步骤如下:
获取 Class 对象的 3 种方式如下:
Persion p = new Persion();
Class clazz = p.getClass();
Class clazz = Persion.class;
Class clazz = Class.forName("fullClassPath"); // fullClassPath 为类的包路径及名称
我们在获得想要操作类的 Class 对象后,可以通过 Class 类中的方法获取并查看该类中的方法和属性,具体的实例代码如下:
// 1.获取 Persion 类的 Class 对象
Class clazz = Class.forName("hello.java.reflect.Persion");
// 2.获取Persion 类的所有方法的信息
Method[] method = clazz.getDeclaredMethods();
for(Method m : method){
System.out.println(m.toString);
}
// 3.获取 Persion 类的所有成员的属性信息
Field[] field = clazz.getDeclaredFields();
for(Field f : field){
System.out.println(f.toString);
}
// 4.获取 Persion 类的所有构造方法的信息
Constructor[] constructor = clazz.getDeclaredConstructors();
for (Constructor c : constructor){
System.out.println(c.toString);
}
创建对象的两种方式如下:
创建对象的具体代码如下:
//1.1:获取 Persion 类的 Class 对象
Class clazz = Class.forName("hello.java.reflect.Persion");
//1.2:使用 newInstance 方法创建对象
Persion p = (Persion) clazz.newInstance();
//2.1:获取构造方法并创建对象
Constructor c = clazz.getDeclaredConstructor(String.class,String.class,int.class);
//2.2:根据构造方法创建对象并设置属性
Persion p1 = (Persion) c.newInstance("李四","男","20");
Method 提供了关于类或接口上某个方法及如何访问该方法的信息,那么在运行代码中如何动态调用该方法呢?答案是通过调用 Method 的 invoke 方法实现。通过 invoke 方法可以实现动态调用,比如可以动态传入参数并将方法参数化。具体过程:获取 Method 对象,并调用 Method 的invoke 方法,如下所述:
//1.获取 Persion 类(hello.java.reflect.Persion)的 Class 对象
Class clz = Class.forName("hello.java.reflect.Persion");
//2.获取 Class 对象中的 setName 方法
Method method = clz.getMethod("setName", String.class);
//3.获取 Constructor 对象
Constructor constructor = clz.getConstructor();
//4.根据 Constructor 定义对象
Object object = constructor.newInstance();
//5.调用 Method 的 invoke 方法,这里的 Method 表示 setName 方法
//因此,相当于动态调用 object 对象的 setName 方法并传入 alex 参数
method.invoke(object,"alex");
以上代码首先通过 Class.forName 方法获取 Persion 类的 Class 对象,然后调用 Persion 类的 Class 对象的 getMethod(“setName”,String.class) 获取一个 Method 对象;接着使用 Class 对象获取指定的 Constructor 对象并调用 Constructor 对象的 newInstance 方法创建 Class 对象对应类的实体;最后调用 method.invoke 方法实现动态调用,这样就通过反射动态生成类的对象并调用其方法。
相关面试题: