美文网首页
Java反射总结

Java反射总结

作者: 追逐地平线的甘 | 来源:发表于2017-08-27 00:27 被阅读0次

1、Class对象的获取

Java中对象可以分为两种,一种是实例对象,一种是Class对象。Class对象是在类加载的时候生成的,而实例对象又是基于Class对象来生成的。
Java中的反射使其拥有了动态语言的属性。在程序开发的过程中,反射技术应用的核心就是Class对象的获取。Class对象的获取有如下三种方法,各有各的使用场景。

public class TestReflect {

    public static void main(String[] args){

        Class<?> clazz1 = null; //Class clazz1 = null;
        Class<?> clazz2 = null; //Class clazz2 = null;
        Class<?> clazz3 = null; //Class clazz3 = null;           
        //使用<?>或者不使用<?>两者是没有区别的,只不过Class支持泛型以后加上<?>会更加规范
        //1、使用Class的静态方法,该方法会产生异常,需要进行异常处理,因为可能没有与所传入字符串相对应的类
        try{

            Class clazz1 = Class.forName("InnerTest");
        } catch (ClassNotFoundException e) {

            e.printStackTrace();
        }

        //2、使用Java中类字面量class获得Class对象
        Class clazz2 = InnerTest.class;

        //3、使用继承于Object的getClass()方法,该方法是native方法
        InnerTest innerTest = new InnerTest();
        Class clazz3 = innerTest.getClass();
    }
}

class InnerTest {

}

2、构造实例对象

获得Class对象之后,采用newInstance(),该方法的返回类型是T,是一个泛型类型。如果Class对象的类型为Class或者Class<?>,则需要强制类型转换。有两种构造实例对象的方法:1、获得Class对象以后调用newInstance()方法;2、获得Constructor对象以后调用newInstance()方法。实际上前者也会调用后者的方法。

import java.lang.reflect.Constructor;
public class TestReflect {

    public static void main(String[] args){

        //若没有传入类型参数,使用newInstance()方法生成对象的时候,会有两种情况:1、只能生成Object对象;2、进行强制类型转换;
        Class<?> clazz1 = null;
        //此处传入了类型参数,使用newInstance()方法生成对象的时候,无需进行强制类型转换;
        Class<InnerTest> clazz2 = null;
        clazz1 = InnerTest.class;
        clazz2 = InnerTest.class;

        //1、使用newInstance()的方法返回实例对象
        try{
            Object test11 = clazz1.newInstance();//只能生成Object对象
            InnerTest test12 = (InnerTest) clazz1.newInstance();//必须进行强制类型转换
            InnerTest test2 = clazz2.newInstance();//直接可以生成所需的类型对象
        }catch (IllegalAccessException e) {
            e.printStackTrace();
        }catch (InstantiationException e){
            e.printStackTrace();
        }

        //2、直接获取构造函数
        Constructor<?> c1 = null;
        Constructor<InnerTest> c2 = null;
        try{
            c1 =clazz1.getDeclaredConstructor();
            c2 = clazz2.getDeclaredConstructor();
        }catch (NoSuchMethodException e) {
            e.printStackTrace();
        }
        try{
            Object innerTest11 = c1.newInstance(null);
            InnerTest innerTest112 = (InnerTest)c1.newInstance(null);
            InnerTest innerTest2 = c2.newInstance(null);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

class InnerTest {

}

3、构造方法

在获得Class的对象以后,通过该对象获得对应的构造函数,如果同时存在多个构造函数。

import java.lang.reflect.Constructor;

public class TestReflect {

    public static void main(String[] args){

        Class<InnerTest> clazz = null;
        clazz = InnerTest.class;
        Constructor<InnerTest> c1 = null;
        Constructor<InnerTest> c2 = null;
        Constructor<InnerTest> c3 = null;
        try{

            c1 = clazz.getDeclaredConstructor();
            c2 = clazz.getDeclaredConstructor(int.class);
            c3 = clazz.getDeclaredConstructor(int.class,int.class);
        }catch (NoSuchMethodException e) {
            e.printStackTrace();
        }
        try{
            InnerTest innerTest1 = c1.newInstance();
            System.out.println("innerTest1 a: " + innerTest1.a + " b: " + innerTest1.b + " c: " + innerTest1.c);
            InnerTest innerTest2 = c2.newInstance(6);
            System.out.println("innerTest2 a: " + innerTest2.a + " b: " + innerTest1.b + " c: " + innerTest2.c);
            InnerTest innerTest3 = c3.newInstance(6,12);
            System.out.println("innerTest3 a: " + innerTest3.a + " b: " + innerTest1.b + " c: " + innerTest3.c);

        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

class InnerTest {
    static int a = 2;

    int b;

    int c;

    public InnerTest() {

    }

    public InnerTest(int b) {
        this.b = b;
    }

    public InnerTest(int b, int c) {
        this.b = b;
        this.c = c;
    }
}

4、非构造方法

非构造方法的获取和使用与构造方法有着明显的区别:1、获取非构造方法不仅需要指定参数的类型,还需要指定方法名;2、非构造方法的调用分为两种情况:a),静态方法调用;b),非静态方法调用

import java.lang.reflect.Method;

public class TestReflect {

    public static void main(String[] args){

        Class<InnerTest> clazz = null;
        clazz = InnerTest.class;
        try{
            //需要指定方法名和参数类型
            Method m1 = clazz.getDeclaredMethod("getA");
            Method m2 = clazz.getDeclaredMethod("setA", int.class);
            //调用静态方法时候,invoke函数的第一个参数为null;
            System.out.println(m1.invoke(null));
            m2.invoke(null,4);
            System.out.println(m1.invoke(null));

            InnerTest innerTest = new InnerTest();
            Method m3 = clazz.getDeclaredMethod("getB");
            Method m4 = clazz.getDeclaredMethod("setB", int.class);
            //调用非静态方法的时候,需要指定对象名;
            System.out.println(m3.invoke(innerTest));
            m4.invoke(innerTest,8);
            System.out.println(m3.invoke(innerTest));
        }catch (Exception e) {
            e.printStackTrace();
        }
    }
}

class InnerTest {
    static int a = 2;

    int b;

    public static int getA() {
        return a;
    }

    public int getB() {
        return b;
    }

    public static void setA(int a) {
        InnerTest.a = a;
    }

    public void setB(int b) {
        this.b = b;
    }
}

5、域

域的获取相对简单,只需要传入变量名字。可以获取域的值和设置域的值。更厉害的时候可以通过setAccessible(),进而访问private的变量

import java.lang.reflect.Field;

public class TestReflect {

    public static void main(String[] args){

        Class<InnerTest> clazz = null;
        clazz = InnerTest.class;
        try{
            Field a = clazz.getDeclaredField("a");
            System.out.println("a: " + a.getInt(null));
            a.setInt(null,4);
            System.out.println("a: " + a.getInt(null));

            InnerTest innerTest = new InnerTest();
            Field b = clazz.getDeclaredField("b");
            System.out.println("b: " + b.getInt(innerTest));
            b.setInt(innerTest,20);
            System.out.println("b: " + b.getInt(innerTest));

            Field c = clazz.getDeclaredField("c");
            System.out.println("accessible: " + c.isAccessible());
            c.setAccessible(true);
            System.out.println("accessible: " + c.isAccessible()+ " c: " +c.getInt(innerTest));
            c.setInt(innerTest,40);
            System.out.println("accessible: " + c.isAccessible()+ " c: " +c.getInt(innerTest));
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

class InnerTest {
    static int a = 2;

    int b = 10;

    private int c = 20;
}

相关文章

  • 反射之一

    总结内容源自一下文章粗浅看java反射机制反射机制应用实践谈谈java反射机制Java Reflection(反射...

  • 反射之二

    总结内容源自一下文章粗浅看java反射机制反射机制应用实践谈谈java反射机制Java Reflection(反射...

  • 反射之三

    总结内容源自以下文章 粗浅看java反射机制 反射机制应用实践 谈谈java反射机制 Java Reflectio...

  • Java反射

    简单总结下什么是Java反射,怎么使用。 Java反射机制 反射就是指JVM运行期,获取指定类所有的属性和方法的一...

  • Java反射使用总结

    Java反射总结 1.反射机制的定义 1.1定义 Java反射机制是在运行状态中,对于任意一个类,都能够知道这个类...

  • java反射总结

    所用到的类 java.lang.Class java.lang.reflect.Constructor java....

  • Java反射总结

    1、Class对象的获取 Java中对象可以分为两种,一种是实例对象,一种是Class对象。Class对象是在类加...

  • Java 反射总结

    什么是反射官方有段介绍是这么说的 Reflection is commonly used by programs ...

  • Java反射总结

    Java反射总结 概述 通常在Java代码中使用一个类,需要在编译时知道类的位置。但是在某些时候,需要使用的类在编...

  • Java反射总结

    概念介绍 Java反射机制JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任...

网友评论

      本文标题:Java反射总结

      本文链接:https://www.haomeiwen.com/subject/vayvdxtx.html