文章目录
标 红 目 录 是 重 点 内 容 ! ! ! 标红目录是重点内容!!! 标红目录是重点内容!!!
1. Java反射机制概述
反射是被视为动态语言的关键,反射机制允许程序在执行期间借助于Reflection API取得任何类的内部信息,并能直接操作任意对象的内部属性及方法。
加载完类之后,在堆内存的方法区就产生了一个Class类型的对象(一个类只有一个Class对象),这个对象就包含了完整的类的结构信息。我们可以通过这个对象看到类的结构。这个对象就像一面镜子,透过这个镜子看到类的结构,所以,我们形象的称之为:反射。
正常方式:引入需要的包类名称 →通过new实例化→取得实例化对象
反射方式:实例化对象→getClass()方法→得到完整的包类名称
1.1 动态语言
是一类在运行时可以改变其结构的语言:例如新的函数、对象、甚至代码可以被引进,已有的函数可以被删除或是其它结构上的变化。通俗点说就是在运行代码时代码可以根据某些条件改变自身结构。
主要动态语言:Object-C、C#、JavaScript、PHP、Python、Erlang。
1.2 静态语言
与动态语言相对应,运行时结构不变的语言就是静态语言。如Java、C、C++。
Java不是动态语言,但Java可以称之为“准动态语言”。即Java有一定的动态性,我们可以利用反射机制、字节码操作获得类似动态语言的特性。Java的动态性让编程的时候更加灵活!
1.3 Java反射机制提供的功能
在运行时判断任意一个对象所属的类;
在运行时构造任意一个类的对象;
在运行时判断任意一个类所具有的成员变量和方法;
在运行时获取泛型信息;
在运行时调用任意一个对象的成员变量和方法;
在运行时处理注解;
生成动态代理。
1.4 反射相关的主要API
Java.lang.Class:代表一个类
Class是用来描述类信息这样的一个类。
Java.lang.reflect.Method:代表类的方法
Java.lang.reflect.Field:代表类的成员变量
Java.lang.reflect.Constructor:代表类的构造器
我们通过下面的代码初步了解一下这几个类:
package reflect;
public class Person {
public String name;
private int age;
public Person() {
}
public Person(String name, int age) {
this.name = name;
this.age = age;
}
private Person(String name){
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
public void show(){
System.out.println("我是Person类!");
}
private void info(){
System.out.println(name);
}
@Override
public String toString() {
return "Person{" +
"name='" + name + '\'' +
", age=" + age +
'}';
}
}
package reflect;
import org.junit.Test;
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
public class ReflectionTest {
//使用反射机制之前,对于Person类我们是如何操作的
@Test
public void test1(){
//1.创建Person类对象
Person person = new Person("Tom", 26);
//2.通过person对象调用其内部的属性/方法
person.setAge(100);
System.out.println(person.toString());
person.show();
//3.在Person类外部,不可以通过person对象调用其内部的私有结构。
//比如说info()方法。
}
//使用反射机制操作Person类
@Test
public void test2() throws Exception{
Class<Person> clazz = Person.class;
//1.通过反射创建Person类对象
Constructor<Person> constructor = clazz.getConstructor(String.class, int.class);
Person jerry = constructor.newInstance("Jerry", 12);
System.out.println(jerry.toString());
//2.通过反射调用对象指定的属性、方法
//调用属性
Field name = clazz.getDeclaredField("name");
name.set(jerry,"another_jerry");//重新设置Jerry对象的name属性
System.out.println(jerry.toString());
//调用方法
Method declaredMethod = clazz.getDeclaredMethod("show");
declaredMethod.invoke(jerry);
/**
* 通过反射可以调用Person类的私有结构
* 比如:私有的构造器、方法、属性
*/
System.out.println("--------------分割线--------------");
Constructor<Person> cons1 = clazz.getDeclaredConstructor(String.class);
cons1.setAccessible(true);
Person jerry1 = cons1.newInstance("jerry");
System.out.println(jerry1);
//调用私有属性
Field name1 = clazz.getDeclaredField("name");
name1.setAccessible(true);
name1.set(jerry1, "hhh");
System.out.println(jerry1);
//调用私有方法
Method info = clazz.getDeclaredMethod("info");
info.setAccessible(true);
info.invoke(jerry1);//相当于jerry1.info();
}
}
看到这里:我们是不是应该有这样的一些疑问:
- 反射机制是不是破坏掉了类的封装性(单例模式)???
- 我要这private有何用?
- 如何看待封装性和反射机制这两个技术?
答疑:
通过直接new的方式或反射的方式都可以调用公共的结构,开发中到底用哪个?我们建议直接用new的方式。那什么时候会使用反射的方式呢,我们在一开始并不知道new哪个类的对象的时候,就可以使用反射机制。
2. 理解Class类并获取Class实例
关于Java.lang.Class类的理解:
- 类的加载过程:程序经过javac.exe命令以后,会生成一个或多个字节码文件(.class后缀结尾)。接着我们使用java.exe命令对某个字节码文件进行解释运行。相当于将某个字节码文件加载到内存中,此过程就称为类的加载。加载到内存当中的类,我们就称为运行时类,此运行时类,就作为Class的一个实例。
怎么去理解这个Class类存在的意义呢,我们可以举一个不太恰当的例子,就是说“概念”这个词也需要一个“概念”去解释它。
Java:万事万物皆对象! - 换句话说,Class的实例就对应着一个运行时类。
- 加载到内存中的运行时类,会缓存一定的时间。在此时间之内,我们可以通过不同的方式来获取此运行时类,我们用不同方式获取的都是唯一的一个运行时类。
/**
* 获取Class实例的方式
*/
@Test
public void test3() throws ClassNotFoundException {
//方式一:调用运行时类的属性→ *.class
Class<Person> personClass1 = Person.class;
//方式二:通过运行时类的对象,调用getClass()
Person person = new Person();
Class<? extends Person> personClass2 = person.getClass();
//方式三:调用Class类的静态方法:forName(String classPath)
Class<?> personClass3 = Class.forName("reflect.Person");
//方式四:使用类加载器(了解即可)
ClassLoader classLoader = ReflectionTest.class.getClassLoader();
Class<?> personClass4 = classLoader.loadClass("reflect.Person");
System.out.println(personClass4);
}
2.1 Class实例对应的结构说明
哪些类型可以有Class对象?
(1)class:外部类、成员(成员内部类,静态内部类),局部内部类,匿名内部类。
(2)interface:接口
(3)[]:数组
(4)enum:枚举
(5)annotation:注解
(6)primitive type:基本数据类型
(7)void:强调:void也算是一种类型!
3. 类的加载与ClassLoader的理解
3.1 类的加载过程
当程序主动使用某个类时,如果该类还未被加载到内存中,则系统会通过如下三个步骤来对该类进行初始化:
加载: 将class文件字节码内容加载到内存中,并将这些静态数据转换成方法区的运行时数据结构,然后生成一个代表这个类的java.lang.class对象,作为方法区中类数据的访问入口(即引用地址)。所有需要访问和使用类数据只能通过这个Class对象。这个加载的过程需要类加载器参与。
链接: 将Java类的二进制代码合并到JVM的运行状态之中的过程。
- 验证:确保加载的类信息符合JVM规范,例如:以cafe开头,没有安全方面的问题
- 准备:正式为类变量(static)分配内存并设置类变量默认初始值的阶段,这些内存都将在方法区中进行分配。
- 解析:虚拟机常量池内的符号引用(常量名)替换为直接引用(地址)的过程。
初始化:
-
执行类构造器()方法的过程。类构造器()方法是由编译期自动收集类中所有类变量的赋值动作和静态代码块中的语句合并产生的。(类构造器是构造类信息的,不是构造该类对象的构造器)。
-
当初始化一个类的时候,如果发现其父类还没有进行初始化,则需要先触发其父类的初始化。
-
虚拟机会保证一个类的()方法在多线程环境中被正确加锁和同步。
package reflect;
public class ReflectionTest {
public static void main(String[] args) {
System.out.println(new A().m);
//输出结果为:200
}
}
class A{
static {
m = 100;
}
static int m = 200;
}
3.2 理解类的加载器
类加载器的作用:
- 类加载的作用:将class文件字节码内容加载到内存中,并将这些静态数据转换成方法区的运行时数据结构,然后在堆中生成一个代表这个类的java.lang.Class对象,作为方法区中类数据的访问入口。
- 类缓存:标准的JavaSE类加载器可以按照要求查找类,但是一旦某个类被加载到类加载器中,它将维持加载(缓存)一段时间。不过JVM垃圾回收机制可以回收这些Class对象。
3.3 了解ClassLoader
类加载器作用是用来把类装载进内存的。JVM规范定义了如下类型的类加载器:
package reflect;
import org.junit.Test;
/**
* 了解类的加载器
*/
public class ClassLoaderTest {
@Test
public void test(){
//对于自定义类,使用【系统类加载器】进行加载
ClassLoader classLoader = ClassLoaderTest.class.getClassLoader();
System.out.println(classLoader);
//调用【系统类加载器】的getParent()方法:获取【扩展类加载器】
ClassLoader classLoader1 = classLoader.getParent();
System.out.println(classLoader1);
//调用【扩展类加载器】的getParent()方法:无法获取【引导类加载器】,返回值为null
//引导类加载器主要负责加载java的核心类库,无法加载自定义类。
ClassLoader classLoader2 = classLoader1.getParent();
System.out.println(classLoader2);
}
}
3.4 例子:使用类加载器读取配置文件
package reflect;
import org.junit.Test;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.util.Properties;
public class ClassLoaderTest {
@Test
/**
* 使用Properties类读取配置文件信息
* 读取配置文件的方式一:直接使用IO流
*/
public void test() throws IOException {
Properties properties = new Properties();
FileInputStream fis = new FileInputStream("src\\reflect\\jdbc.properties");
properties.load(fis);
String username = properties.getProperty("username");
String password = properties.getProperty("password");
fis.close();
System.out.println(username);
System.out.println(password);
}
@Test
/**
* 读取配置文件的方式二:使用类的加载器
*/
public void test2() throws IOException {
Properties properties = new Properties();
ClassLoader classLoader = ClassLoaderTest.class.getClassLoader();
//配置文件默认识别为:当前module的src目录下
InputStream is = classLoader.getResourceAsStream("reflect\\jdbc.properties");
properties.load(is);
String username = properties.getProperty("username");
String password = properties.getProperty("password");
is.close();
System.out.println(username);
System.out.println(password);
}
}
4. 创建运行时类的对象
package reflect;
import org.junit.Test;
import java.util.Random;
/**
* 通过反射创建对应的运行时类的对象
* 体会反射的动态性
*/
public class NewInstanceTest {
@Test
public void test() throws IllegalAccessException, InstantiationException, ClassNotFoundException {
int num = new Random().nextInt(3);
String classpath = null;
switch (num){
case 0: classpath = "java.util.Date";
break;
case 1: classpath = "java.lang.String";
break;
case 2: classpath = "reflect.Person";
break;
}
Object instance = getInstance(classpath);
System.out.println(instance);
}
public Object getInstance(String classpath) throws ClassNotFoundException, IllegalAccessException, InstantiationException {
Class<?> clazz = Class.forName(classpath);
return clazz.newInstance();
}
}
5. 获取运行时类的完整结构
以Person类为例说明:
1.创建Person类的父类Creature:
package ceshi;
import java.io.Serializable;
public class Creature<T> implements Serializable {
private char gender;
public double weight;
private void breath(){
System.out.println("生物呼吸");
}
public void eat(){
System.out.println("生物饮食");
}
}
2.创建自定义接口和自定义注解
package ceshi;
public interface MyInterface {
void info();
}
package ceshi;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
import static java.lang.annotation.ElementType.*;
import static java.lang.annotation.ElementType.CONSTRUCTOR;
import static java.lang.annotation.ElementType.LOCAL_VARIABLE;
@Target({TYPE, FIELD, METHOD, PARAMETER, CONSTRUCTOR, LOCAL_VARIABLE})
@Retention(RetentionPolicy.RUNTIME)
public @interface MyAnnotation {
String value() default "hello";
}
3.创建Person类
package ceshi;
@MyAnnotation
public class Person extends Creature<String> implements Comparable<String>,MyInterface{
private String name;
int age;
public int id;
public Person(){}
@MyAnnotation(value = "abc")
private Person(String name){
this.name = name;
}
Person(String name,int age){
this.name = name;
this.age = age;
}
@MyAnnotation
private String show(String nation){
System.out.println("我的国籍:"+nation);
return nation;
}
@Override
public int compareTo(String o) {
return 0;
}
public void setName(String name) {
this.name = name;
}
public void setAge(int age) {
this.age = age;
}
public void setId(int id) {
this.id = id;
}
@Override
public void info() {
System.out.println("人类");
}
public String getName() {
return name;
}
public int getAge() {
return age;
}
public int getId() {
return id;
}
private static void showDesc(){
System.out.println("我是一个可爱的人!(*^▽^*)");
}
@Override
public String toString() {
return "Person{" +
"name='" + name + '\'' +
", age=" + age +
", id=" + id +
'}';
}
}
5.1 获取运行时类的属性结构及其内部结构
package ceshi;
import org.junit.Test;
import java.lang.reflect.Field;
import java.lang.reflect.Modifier;
public class FieldTest {
@Test
public void test1(){
Class<Person> personClass = Person.class;
//获取属性结构
//getFields():获取当前运行时类及其父类中声明为public访问权限的属性
Field[] fields = personClass.getFields();
for(Field f:fields)
System.out.println(f);
//getDeclaredFields():获取当前运行时类中声明的所有属性,不包含父类中的属性
Field[] declaredFields = personClass.getDeclaredFields();
for(Field f:declaredFields)
System.out.println(f);
}
@Test
public void test2(){
Class<Person> personClass = Person.class;
Field[] declaredFields = personClass.getDeclaredFields();
for(Field f:declaredFields){
//1.获取权限修饰符
int modifiers = f.getModifiers();
System.out.println(Modifier.toString(modifiers));
//2.获取数据类型
Class<?> type = f.getType();
//3.获取变量名(属性名)
String name = f.getName();
}
}
}
5.2 获取运行时类的方法结构
package ceshi;
import org.junit.Test;
import java.lang.reflect.Method;
public class MethodTest {
@Test
public void test(){
Class<Person> personClass = Person.class;
//getMethods():获取当前运行时类及其所有父类中声明为public权限的方法
Method[] methods = personClass.getMethods();
for(Method method:methods)
System.out.println(method);
System.out.println();
//getDeclaredMethods():获取当前运行时类中声明的方法(不包含父类中声明的方法)
Method[] declaredMethods = personClass.getDeclaredMethods();
for(Method method:declaredMethods)
System.out.println(method);
}
}
5.3 获取运行时类的方法的内部结构
package ceshi;
import org.junit.Test;
import java.lang.annotation.Annotation;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
public class MethodTest {
@Test
public void test1(){
Class<Person> personClass = Person.class;
Method[] declaredMethods = personClass.getDeclaredMethods();
for(Method method:declaredMethods){
//1.获取方法声明的注解
Annotation[] annotations = method.getAnnotations();
for(Annotation annotation:annotations)
System.out.println(annotation);
//2.权限修饰符
int modifiers = method.getModifiers();
System.out.println(Modifier.toString(modifiers));
//3.返回值类型
System.out.println(method.getReturnType());
//4.方法名
System.out.print(method.getName());
//5.形参列表
Class<?>[] parameterTypes = method.getParameterTypes();
if(!(parameterTypes==null&¶meterTypes.length==0)){//没有参数类型
for(Class<?> parameterType:parameterTypes){
System.out.println(parameterType.getName());
}
}
//6.抛出的异常
Class<?>[] exceptionTypes = method.getExceptionTypes();
for(Class<?> exceptionType:exceptionTypes)
System.out.println(exceptionType.getName());
}
}
}
5.4 获取运行时类的构造器结构
package ceshi;
import org.junit.Test;
import java.lang.reflect.Constructor;
public class ConstructorTest {
@Test
public void test(){
Class<Person> personClass = Person.class;
//getConstructors():获取当前运行时类中声明为public的构造器
Constructor<?>[] constructors = personClass.getConstructors();
for(Constructor<?> constructor:constructors)
System.out.println(constructor);
//getDeclaredConstructors():获取当前运行时类中声明的所有构造器
Constructor<?>[] declaredConstructors = personClass.getDeclaredConstructors();
for(Constructor<?> declaredConstructor:declaredConstructors)
System.out.println(declaredConstructor);
}
}
5.5 获取运行时类的父类及父类的泛型
package ceshi;
import org.junit.Test;
import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;
public class ParentTest {
@Test
public void test(){
Class<Person> personClass = Person.class;
//获取运行时类的父类
Class<? super Person> superclass = personClass.getSuperclass();
System.out.println(superclass);
//获取运行时类的父类(带泛型的父类)
Type genericSuperclass = personClass.getGenericSuperclass();
System.out.println(genericSuperclass);
//获取运行时类的带泛型的父类的泛型
ParameterizedType paramType = (ParameterizedType)genericSuperclass;
//获取泛型
Type[] actualTypeArguments = paramType.getActualTypeArguments();
for(Type actualTypeArgument : actualTypeArguments)
System.out.println(actualTypeArgument.getTypeName());
}
}
5.6 获取运行时类的接口、所在包、注解
package ceshi;
import org.junit.Test;
import java.lang.annotation.Annotation;
public class InterfaceTest {
@Test
public void test(){
Class<Person> personClass = Person.class;
//获取注解
Class<?>[] interfaces = personClass.getInterfaces();
for(Class<?> i: interfaces)
System.out.println(i);
//获取所在包
Package cur_package = personClass.getPackage();
System.out.println(cur_package);
//获取注解
Annotation[] annotations = personClass.getAnnotations();
for(Annotation annotation:annotations)
System.out.println(annotation);
}
}
6. 调用运行时类的指定结构
6.1 调用运行时类中的指定属性
package reflect;
import ceshi.Person;
import org.junit.Test;
import java.lang.reflect.Field;
public class ReflectionTest {
@Test
public void test() throws Exception {
Class<Person> personClass = Person.class;
Person person = personClass.newInstance();
//获取指定的属性:要求运行时类中属性声明为public
//通常不用这个,用getDeclaredField()
Field id = personClass.getField("id");
//设置当前属性的值
id.set(person, 123);
//获取当前属性的值
//get():参数1为获取哪个对象的当前属性值
Object val = id.get(person);
System.out.println(val);
}
}
6.2 调用运行时类中的指定方法
package reflect;
import ceshi.Person;
import org.junit.Test;
import java.lang.reflect.Method;
public class ReflectionTest {
@Test
public void test() throws Exception {
Class<Person> personClass = Person.class;
Person person = personClass.newInstance();
//获取指定的某个方法
//getDeclaredMethod():参数1:指明获取的方法的名称;参数2:指明获取的方法的形参列表。
Method show = personClass.getDeclaredMethod("show", String.class);
//invoke():参数1:方法的调用者;参数2:给方法形参赋值的实参
show.setAccessible(true);
show.invoke(person,"调用Person类的show()方法");
//invoke()方法的返回值即为对应类中调用的方法的返回值。
}
/**
*如何调用静态方法
*/
@Test
public void test1() throws Exception {
Class<Person> personClass = Person.class;
Method showDesc = personClass.getDeclaredMethod("showDesc");
showDesc.setAccessible(true);
showDesc.invoke(personClass);
}
}
6.3 调用运行时类中的指定构造器
package reflect;
import org.junit.Test;
import ceshi.Person;
import java.lang.reflect.Constructor;
public class ReflectionTest {
@Test
public void test() throws Exception {
Class<Person> personClass = Person.class;
//1.获取指定的构造器
Constructor<Person> constructor = personClass.getDeclaredConstructor(String.class);
//2.保证此构造器是可以访问的
constructor.setAccessible(true);
//3.调用此构造器创建运行时类的对象
Person person = constructor.newInstance("Tom");
System.out.println(person);
}
}