前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >Java 反射(Relfect)

Java 反射(Relfect)

原创
作者头像
HLee
修改2021-09-13 10:09:43
8050
修改2021-09-13 10:09:43
举报
文章被收录于专栏:房东的猫房东的猫

简介

反射 (Reflection) 是 Java 的特征之一,它允许运行中的 Java 程序获取自身的信息,并且可以操作类或对象的内部属性。

简而言之,通过反射,我们可以在运行时获得程序或程序集中每一个类型的成员和成员的信息。程序中一般的对象的类型都是在编译期就确定下来的,而 Java反射机制可以动态地创建对象并调用其属性,这样的对象的类型在编译期是未知的。所以我们可以通过反射机制直接创建对象,即使这个对象的类型在编译期是未知的。

反射的核心是JVM在运行时才动态加载类或调用方法/访问属性,它不需要事先(写代码的时候或编译期)知道运行对象是谁。

Java 反射主要提供以下功能:

  • 在运行时判断任意一个对象所属的类;
  • 在运行时构造任意一个类的对象;
  • 在运行时判断任意一个类所具有的成员变量和方法(通过反射甚至可以调用private方法);
  • 在运行时调用任意一个对象的方法;

由于反射会额外消耗一定的系统资源,因此如果不需要动态地创建一个对象,那么就不需要用反射。

主要用途

很多人都认为反射在实际的Java开发应用中并不广泛,其实不然。当我们在使用 IDE(如 Eclipse,IDEA)时,当我们输入一个对象或类并想调用它的属性或方法时,一按点号,编译器就会自动列出它的属性或方法,这里就会用到反射。

反射最重要的用途就是开发各种通用框架很多框架(比如 Spring)都是配置化的(比如通过 XML 文件配置 Bean),为了保证框架的通用性,它们可能需要根据配置文件加载不同的对象或类,调用不同的方法,这个时候就必须用到反射,运行时动态加载需要加载的对象。

基本运用

反射相关的类一般都在java.lang.relfect包里。

获得 Class 对象

  • 使用Class类的forName静态方法:
代码语言:javascript
复制
Class.forName(driver);
在 JDBC 开发中常用此方法加载数据库驱动
  • 直接获取某一个对象的class
代码语言:javascript
复制
Class<?> klass = int.class;
Class<?> classInt = Integer.TYPE;
  • 调用某个对象的getClass()方法
代码语言:javascript
复制
StringBuilder str = new StringBuilder("123");
Class<?> klass = str.getClass();

判断是否为某个类的实例

一般地,我们用instanceof关键字来判断是否为某个类的实例。同时我们也可以借助反射中Class对象的isInstance()方法来判断是否为某个类的实例,它是一个 native 方法:

代码语言:javascript
复制
public native boolean isInstance(Object obj);

例:
StringBuilder str = new StringBuilder("123");
Class<?> klass = str.getClass();

if (klass.isInstance(new StringBuilder())) {
    System.out.println("true");
}
System.out.println(klass);

true
class java.lang.StringBuilder

创建实例

通过反射来生成对象主要有两种方式。

  • 使用Class对象的newInstance()方法来创建Class对象对应类的实例。
代码语言:javascript
复制
Class<?> c = String.class;
Object str = c.newInstance();
  • 先通过Class对象获取指定的Constructor对象,再调用Constructor对象的newInstance()方法来创建实例。这种方法可以用指定的构造器构造类的实例。
代码语言:javascript
复制
//获取String所对应的Class对象
Class<?> c = String.class;
//获取String类带一个String参数的构造器
Constructor constructor = c.getConstructor(String.class);
//根据构造器创建实例
Object obj = constructor.newInstance("23333");
System.out.println(obj);

获取方法

获取某个Class对象的方法集合,主要有以下几个方法:

  • getDeclaredMethods 方法返回类或接口声明的所有方法,包括公共、保护、默认(包)访问和私有方法,但不包括继承的方法。
代码语言:javascript
复制
public Method[] getDeclaredMethods() throws SecurityException {}

StringBuilder str = new StringBuilder("123");
Class<?> klass = str.getClass();
klass.getDeclaredMethods();
  • getMethods 方法返回某个类的所有公用(public)方法,包括其继承类的公用方法。
代码语言:javascript
复制
public Method[] getMethods() throws SecurityException {}
  • getMethod 方法返回一个特定的方法,其中第一个参数为方法名称,后面的参数为方法的参数对应Class的对象。
代码语言:javascript
复制
public Method getMethod(String name, Class<?>... parameterTypes) {}

获取构造器信息

获取类构造器的用法与上述获取方法的用法类似。主要是通过Class类的getConstructor方法得到Constructor类的一个实例,而Constructor类有一个newInstance方法可以创建一个对象实例。

获取类的成员变量(字段)信息

  • getFiled:访问公有的成员变量
  • getFileds
  • getDeclaredField:所有已声明的成员变量,但不能得到其父类的成员变量
  • getDeclaredFields

调用方法

当我们从类中获取了一个方法后,我们就可以用invoke()方法来调用这个方法。invoke方法的原型为:

代码语言:javascript
复制
public Object invoke(Object obj, Object... args)
        throws IllegalAccessException, IllegalArgumentException,
           InvocationTargetException {}
代码语言:javascript
复制
public class test1 {
    public static void main(String[] args) throws IllegalAccessException, InstantiationException, NoSuchMethodException, InvocationTargetException {
        Class<?> klass = methodClass.class;
        //创建methodClass的实例
        Object obj = klass.newInstance();
        //获取methodClass类的add方法
        Method method = klass.getMethod("add", int.class, int.class);
        //调用method对应的方法 => add(1,4)
        Object result = method.invoke(obj, 1, 4);
        System.out.println(result);
    }
}

class methodClass {

    public final int fuck = 3;
    
    public int add(int a,int b) {
        return a + b;
    }
    
    public int sub(int a,int b) {
        return a + b;
    }
}

利用反射创建数组

数组在Java里是比较特殊的一种类型,它可以赋值给一个Object Reference。下面我们看一看利用反射创建数组的例子:

代码语言:javascript
复制
public static void testArray() throws ClassNotFoundException {
     Class<?> cls = Class.forName("java.lang.String");
     Object array = Array.newInstance(cls, 25);
     //往数组里添加内容
     Array.set(array, 0, "hello");
     Array.set(array, 1, "Java");
     Array.set(array, 2, "fuck");
     Array.set(array, 3, "Scala");
     Array.set(array, 4, "Clojure");
     
     //获取某一项的内容
     System.out.println(Array.get(array, 3));
}

原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。

如有侵权,请联系 cloudcommunity@tencent.com 删除。

原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。

如有侵权,请联系 cloudcommunity@tencent.com 删除。

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 简介
    • 主要用途
    • 基本运用
      • 获得 Class 对象
        • 判断是否为某个类的实例
          • 创建实例
            • 获取方法
              • 获取构造器信息
                • 获取类的成员变量(字段)信息
                  • 调用方法
                    • 利用反射创建数组
                    领券
                    问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档