Java基础-Java反射


前言
1. 动态语言的概念

动态语言是指程序在运行时可以改变其结构:新的函数可以引进,已有的函数可以被删除等结构上的变化。
比如常见的JavaScript就是动态语言,除此之外Ruby、Python等也属于动态语言,而C、C++则不属于动态语言。
从反射角度来说Java属于半动态语言

2. 反射机制概念

在Java中的反射机制是指在运行状态中,对于任意一个类都能够知道这个类所有的属性和方法,并且对于任意一个对象,都能够调用它的任意一个方法,这种动态获取信息以及动态调用对象方法的功能称为Java的反射机制。

3. 反射的应用场合
3.1. 编译时类型和运行时类型

在Java程序中许多对象在运行时都会出现两种类型:编译时类型和运行时类型。
编译时的类型由声明对象时使用的类型来决定,运行时类型由实际赋值给对象的类型决定。如:Person p = new Student();
其中编译时类型为Person,运行时类型为Student。

3.2. 编译时类型无法获取具体方法

程序在运行时还可能接收到外部传入的对象,该对象的编译时类型为Object,但是程序有需要调用该对象的运行时类型的方法。为了解决这些问题,程序需要在运行时发现对象和类的真实信息。然而,如果编译时根本无法预知该对象和类属于哪些类,程序只能依靠运行时信息来发现该对象和类的真实信息,此时就必须使用反射。

Java反射API
1. 相关类
1.1. Class类

反射的核心类,可以获取类的属性、方法等信息

  • 获得类相关的方法
方法用途
asSubclass(Class clazz)把传递的类的对象转换成代表其子类的对象
Cast把对象转换成代表类或是接口的对象
getClassLoader()获得类的加载器
getClasses()返回一个数组,数组中包含该类中所有公共类和接口类的对象
getDeclaredClassed()返回一个数组,数组中包含该类中所有类和接口类的对象
forName(String className)根据类名返回类的对象
getName()获得类的完整路径名字
newInstance()创建类的实例
getPackage()获取类的包
getSimpleName()获取类的名字
getSuperclass()获得当前类继承的父类的名字
getInterfaces()获得当前类实现的类或是接口
getField(String name)获得某个共有的数据对象
  • 获得类中属性相关的方法
方法用途
getFields()获得所有公有的属性对象
getDeclaredField(String name)获得某个属性对象
getDeclaredFields()获得所有属性对象
  • 获得类中构造器相关的方法
方法用途
getConstructor(Class …<?> parameterTypes)获得该类中与参数类型匹配的共有构造方法
getConstructors()获得该类的所有公有构造方法
getDeclaredConstructor(Class…<?> parameterTypes获得该类中与参数类型匹配的构造方法
getDeclaredConstructors()获得该类所有构造方法
  • 获得类中方法相关的方法
方法用途
getMethod(String name, Class…<?> parameterTypes获得该类某个公有的方法
getMethods()获得该类所有公有的方法
getDeclaredMethod(String name, Class…<?> parameterTypes)获得该类某个方法
getDeclaredMethods()获得该类所有方法
  • 还有其它许多方法此处省略… 想了解的朋友可以去看一下源码
1.2. Field类

java.lang.reflec包中的类,表示类的成员变量,可以用来获取和设置类之中的属性值

方法用途
equals(Object obj)属性与对象相等则返回true
get(Object obj)获得对象中对应的属性值
set(Object obj, Object value)设置对象中对应的属性值
1.3. Method类

java.lang.reflec包中的类,表示类的方法,它可以用来获取类中的方法信息或者执行方法

方法用途
invoke(Object obj, Object … args)传递obj对象及参数调用该对象对应的方法
getParameterCount获得方法参数数量
1.4. Constructor类

java.lang.reflec包中的类,表示类的构造方法

方法用途
newInstance(Object … iniitargs)根据传递的参数创建类的对象
getParameterCount获得方法参数数量
2. 反射使用步骤
  • 获取想要操作类的Class对象,它是反射的核心,通过Class对象我们可以任意调用类的方法
  • 调用Class类中的方法,就是反射的使用阶段
  • 使用反射API来操作这些信息
3. 获取Class对象的3种方法
//调用某个对象getClass()方法
Student s = new Student();
Class clazz = s.getClass();

//调用某个类的class属性来获取该类的对应的Class对象
Class clazz = Student.class;

//使用Class类中的forName()静态方法
Class clazz = Class.forName("类的全路径");

当我们获得了想要操作的类的Class对象后,可以通过Class类种的方法获取并查看该类中的方法和属性。
假设我此时有一个Student类:

package com.soul.fanshe;

/**  
 * @Title: Student.java
 * @Description: TODO(描述)
 * @author Soul
 * @date 2020-05-23 11:10:31 
 */
public class Student {
      private String name;
      private String hight;
      private Integer age;

      public Student() {    super();  }
      public Student(String name, String hight, Integer age) {
            super();
            this.name = name;
            this.hight = hight;
            this.age = age;
      }
      
      public String getName() { return name; }
      public void setName(String name) {this.name = name;}
      public String getHight() { return hight; }
      public void setHight(String hight) {this.hight = hight;}
      public Integer getAge() { return age;}
      public void setAge(Integer age) {this.age = age;}
}

我们再写一个测试类,来获取Student类中的相关信息,代码如下:

// 获取Student类的Class对象
Class clazz = Class.forName("com.soul.fanshe.Student");

// 获取Student类的所有方法信息
Method [] method = clazz.getDeclaredMethods();
System.out.println(clazz.getName() + "类的所有方法:");
for(Method m:method){
    System.out.println(m.toString());
}

// 获取Student类的所有成员属性信息
Field[] field = clazz.getDeclaredFields();
System.out.println(clazz.getName() + "类的所有成员属性信息:");
for(Field f : field){
    System.out.println(f.toString());
}

// 获取Student类的所有构造方法信息
Constructor [] constructor = clazz.getDeclaredConstructors();
System.out.println(clazz.getName() + "类的所有构造方法信息:");
for(Constructor c : constructor){
    System.out.println(c.toString());
}

/** 
* 输入结果如下
**/
com.soul.fanshe.Student类的所有方法:
public java.lang.String com.soul.fanshe.Student.getName()
public void com.soul.fanshe.Student.setName(java.lang.String)
public void com.soul.fanshe.Student.setAge(java.lang.Integer)
public void com.soul.fanshe.Student.setHight(java.lang.String)
public java.lang.String com.soul.fanshe.Student.getHight()
public java.lang.Integer com.soul.fanshe.Student.getAge()

com.soul.fanshe.Student类的所有成员属性信息:
private java.lang.String com.soul.fanshe.Student.name
private java.lang.String com.soul.fanshe.Student.hight
private java.lang.Integer com.soul.fanshe.Student.age

com.soul.fanshe.Student类的所有构造方法信息:
public com.soul.fanshe.Student()
public com.soul.fanshe.Student(java.lang.String,java.lang.String,java.lang.Integer)
4. 创建对象的两种方法
4.1. Class对象的newInstance()

使用Class对象的newInstance()方法来创建该Class对象对应类的实例,但是这种方法要求该Class对象对应的类有默认的空构造器。

Class sClazz = Class.forName("com.soul.fanshe.Student");

// 使用newInstance方法创建对象
// 如果创建实例的类无空构造器,则会抛出异常
Student s1 = (Student) sClazz.newInstance();
4.2. Constructor对象newInstance()方法

先使用Class对象获取指定的Constructor对象,再调用Constructor对象的newInstance()方法来创建Class对象对应类的实例,通过这种方法可以选定构造方法创建实例。

// 获取构造方法创建对象
Constructor c = sClazz.getDeclaredConstructor(String.class, String.class, Integer.class);
// 创建对象并设置属性
Student s2 = (Student) c.newInstance("张三","185", 20);
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值