java 反射

一:什么是java发射

反射(Reflection)是Java程序开发语言的特性之一,它允许运行中的Java程序获得自身信息,并可以操作类和对象的内部属性。简而言之,通过反射,我们可以获取到运行时程序集中每一个类型的成员和成员的信息。程序中一般的对象类型都是在编译期就确定下来的,而java 反射机制可以动态的创建对象和调用其属性,这样的对象类型在编译期间是位置的。所以我们通过反射机制来创建对象,即使这个对象在编译期间是未知的。 反射的核心是jvm运行时才动态的加载类或调用方法/访问属性,它不需要事先(写代码的时候或编译期)知道运行对象是谁。

java反射框架主要提供以下功能:

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

二:反射的应用

反射的应用:反射可以应用于判断任意对象所属的类,获得Class对象,构造任意一个对象。

  1. 获得Class对象
    方法有三种
    (1)使用Class类的forName静态方法
public static Class<?> forName(String className)

Class.forName(devier) ;// jdbc 中常用此方法加载数据库驱动

(2) 直接获取木一个对象的class,比如:

Class<?> Klass = int.class;
Class<?> classInt = Integer.TYPE;

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

String str = new String("ccc");
Class<?> klass = str.getClass();
  1. 判断是否为某个对象的class,比如:
    一般地,我们用instanceof关键字来判断是否时某个类的实例。同时我们使用反射中Class对象的isInstance()方法来判断是否为某个类的实例,这是一个native方法:
public native boolean isInstance(Object obj);
  1. 创建实例
    通过反射来创建对象主要有俩种方式
    (1) 使用Class对象的newInstance()方法来创建Class对象对应类的实例
Class<?> c = String.class;
Object str = c.newInstance();

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

Class<?> c = String.class;   //获取String所对应的Class对象
Constructor constructor = c.getConstructor(String.class); //获取String 类一个带参数的构造器
Object obj = constructor.newInstance("reflection");  //根据构造器创建实例
System.out.println(obj);
  1. 获取方法

    获取某个Class对象的方法集合,主要有以下几个方法:
    getDeclaredMethods() 方法返回类或接口生命的所有方法,包括public,proteced,private,但不包括继承的方法。

public Method[] getDeclaredMethods() throws SecurityException

getMethod()方法返回某个类所有公用(public)方法,包括继承类的公用方法

public Method[] getMethods() throws SecurityException

getMethod 方法返回一个特定的方法,其中一个参数方法名称,后面的参数为方法的参数对象Class的对象

public Method getMethod(String name,Class<?>... param)

举例

import java.lang.reflect.Method;

/**
 * Created by qiaozhiquan on 2018/5/9.
 */
public class ReflectionDemo {

    public static void main(String [] args) throws Exception {
        Demo demo = new Demo();
        Class<?> klass = demo.getClass();
        Class<?> klass1 = Demo.class;
        Class<?> klass2 = Class.forName("demo.Demo");

        Object object = klass.newInstance();
        Method[] methods = klass.getMethods();
        Method[] declaredMethods = klass1.getDeclaredMethods();
        Method method = klass.getMethod("add",int.class,int.class);
        System.out.println("getMethods 获取的方法:");
        for(Method m:methods){
            System.out.println(m);
        }

        System.out.println("getDeclaredMethods获取的方法:");
        for(Method m:declaredMethods) {
            System.out.println(m);
        }

        System.out.println(method.invoke(object,1,2));

    }

}

class Demo {

    private int start = 0;
    private int end = 10;

    public int add(int a, int b) {
        return a + b;
    }

    public int sub(int c, int d) {
        return c - d;
    }

}

运行结果如下:

getMethods 获取的方法:
public int demo.Demo.add(int,int)
public int demo.Demo.sub(int,int)
public final void java.lang.Object.wait() throws java.lang.InterruptedException
public final void java.lang.Object.wait(long,int) throws java.lang.InterruptedException
public final native void java.lang.Object.wait(long) throws java.lang.InterruptedException
public boolean java.lang.Object.equals(java.lang.Object)
public java.lang.String java.lang.Object.toString()
public native int java.lang.Object.hashCode()
public final native java.lang.Class java.lang.Object.getClass()
public final native void java.lang.Object.notify()
public final native void java.lang.Object.notifyAll()
getDeclaredMethods获取的方法:
public int demo.Demo.add(int,int)
public int demo.Demo.sub(int,int)
3

可以看到,通过getMethods()获取的方法可以获取到父类的方法,比如java.lang.Object下定义的各种方法。

  1. 获取构造器信息
    获取构造器类的用法与上述的用法类似。主要时通过Class类的getConstructor方法得到Constructor类的一个实例,而Constructor类有一个newInstance方法可以创建一个对象实例:
public T newInstance(Object ... initargs); //此方法可以根据传入的参数来调用Constructor创建对象实例~
  1. 获取类的成员变量(字段)信息
    getField:访问共有的成员你变量
    getDeclaredField:所有已声明的成员变量。但不能得到其父类的成员变量
    getFields和getDeclaredFields用法同上(参照Method)

    1. 调用方法
      当我们从类中获取了一个方法以后,我们就可以调用这个方法。invoke方法的原型为:
@CallerSensitive
    public Object invoke(Object obj, Object... args)
        throws IllegalAccessException, IllegalArgumentException,
           InvocationTargetException
  1. 利用反射创建数组

数组时一种特殊的数据类型,它可以直接赋值给一个ObjectReference。

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));
    }

其中Array类为java.lang.reflect.Array类,我们通过Array.newInstance()来创建数组对象,它的原型是:

public static Object newInstance(Class<?> componentType, int length)
        throws NegativeArraySizeException {
        return newArray(componentType, length);
    }
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值