Java反射初体验

要说Java反射,首先说一下动态语言,了解了动态语言的特性方便我们更好的理解反射。

动态语言

动态语言是指在运行期间内可以改变其结构的语言,例如新的函数可以被引进,已有的函数可以在结构上发生变化。像典型的C#、JavaScript、Python等。

优点:方便阅读,清晰明了。

缺点:不方便调试。

Java并不是动态语言,但有了反射机制,Java可以称之为半动态语言。

反射

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

反射的应用场景

在 Java 程序中许多对象在运行是都会出现两种类型:编译时类型和运行时类型。 编译时的类型由声明对象时实用的类型来决定,运行时的类型由实际赋值给对象的类型决定 。

Person person = new Student();
//其中编译时类型为 Person,运行时类型为 Student。

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

Java反射API

反射 API 用来生成 JVM 中的类、接口或则对象的信息。

  1. Class 类:反射的核心类,可以获取类的属性,方法等信息。
  2. Field 类:Java.lang.reflec 包中的类,表示类的成员变量,可以用来获取和设置类之中的属性
    值。
  3. Method 类: Java.lang.reflec 包中的类,表示类的方法,它可以用来获取类中的方法信息或
    者执行方法。
  4. Constructor 类: Java.lang.reflec 包中的类,表示类的构造方法。
反射的使用步骤

调用某个对象的 getClass()方法

调用某个类的 class 属性来获取该类对应的 Class 对象

使用 Class 类中的 forName()静态方法(最安全/性能最好)

public class Person {
    public String name;
    private int age;

    public Person() {
    }

    public Person(String name, int age) {
        this.name = name;
        this.age = age;
    }

    public void eat(){
        System.out.println("this is public method!");
    }
    private void dance(){
        System.out.println("this is private method!");
    }
}

通过反射获取类的信息

public class Test {
    @org.junit.Test
    public void test1() throws ClassNotFoundException {
        Person p = new Person();
        //获取类信息
        //调用某个对象的 getClass()方法
        Class clazz = p.getClass();
        //调用某个类的 class 属性来获取该类对应的 Class 对象
        Class clazz1 = Person.class;
        //使用 Class 类中的 forName()静态方法(最安全/性能最好)
        Class clazz2 = Class.forName("com.xzy.reflect.Person");
        System.out.println(clazz);
        System.out.println(clazz1);
        System.out.println(clazz2);
        System.out.println(clazz == clazz1);
        System.out.println(clazz == clazz2);

    }
}
运行结果:
class com.xzy.reflect.Person
class com.xzy.reflect.Person
class com.xzy.reflect.Person
true
true

当我们获得了想要操作的类的 Class 对象后,可以通过 Class 类中的方法获取并查看该类中的方法和属性。

public class Test {
    @org.junit.Test
    public void test2() throws ClassNotFoundException {
        Class clazz = Class.forName("com.xzy.reflect.Person");
        //获取Person类的所有方法
        Method[] declaredMethods = clazz.getDeclaredMethods();

        System.out.println("获取到的所有方法:");
        for (Method methods : declaredMethods) {
            System.out.println(methods.toString());
        }

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

        //获取 Person 类的所有构造方法信息
        Constructor[] constructor = clazz.getDeclaredConstructors();
        System.out.println("获取到的所有构造器:");
        for (Constructor c : constructor) {
            System.out.println(c.toString());
        }
    }
}
运行结果:
获取到的所有方法:
public void com.xzy.reflect.Person.eat()
private void com.xzy.reflect.Person.dance()
获取到的所有属性:
public java.lang.String com.xzy.reflect.Person.name
private int com.xzy.reflect.Person.age
获取到的所有构造器:
public com.xzy.reflect.Person()
public com.xzy.reflect.Person(java.lang.String,int)

Process finished with exit code 0

创建对象的两种方法
Class 对象的 newInstance()

  1. 使用 Class 对象的 newInstance()方法来创建该 Class 对象对应类的实例,但是这种方法要求该 Class 对象对应的类有默认的空构造器。调用 Constructor对象的 newInstance()
  2. 先使用 Class 对象获取指定的 Constructor 对象,再调用 Constructor 对象的 newInstance()方法来创建 Class 对象对应类的实例,通过这种方法可以选定构造方法创建实例。
    @org.junit.Test
    public void test3() throws Exception {
        //获取 Person 类的 Class 对象
        Class clazz = Class.forName("com.xzy.reflect.Person");
        //使用.newInstance 方法创建对象
        Person p = (Person) clazz.newInstance();
        //获取构造方法并创建对象
        Constructor c = clazz.getDeclaredConstructor(String.class, int.class);
        //创建对象并设置属性
        Person p1 = (Person) c.newInstance("张三", 20);
        System.out.println(p);
        System.out.println(p1);
    }
}

通过invoke()调用具体方法

method.invoke(obj,args);
obj:具体的实例对象
args:参数的类型,可以是数组

如果是私有方法,一定要打开私有方法的访问权限:

//重点:此处打开私有方法的访问权限,否则会报错
dance.setAccessible(true); 
    @org.junit.Test
    public void test4() throws Exception {
        //获取 Person 类的 Class 对象
        Class clazz = Class.forName("com.xzy.reflect.Person");

        //使用.newInstance 方法创建对象
        Person p = (Person) clazz.newInstance();

        //如果有参数可以在名字后面指定哪些参数的类型
        Method eat = clazz.getDeclaredMethod("eat");
        Method dance = clazz.getDeclaredMethod("dance");

        //重点:此处打开私有方法的访问权限,否则会报错
        dance.setAccessible(true);

        eat.invoke(p);
        dance.invoke(p);

    }
}
运行结果:
this is public method!
this is private method!

Process finished with exit code 0

以上仅为反射简单使用,具体请参考官方文档。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值