java基础整理02--反射的基础知识点

1、Class类是啥?

java程序中的各个java类是属于同一类事物,描述这类事物的java类名就是Class。

所以说众多的java类用Class表示。
Class类代表了java类。
Class的实例怎么获得呢?Class实例代表了啥呢 ?

Class实例可以用三种方式获得,每一个实例代表一份字节码文件:
下面是获得Class实例的三种实现方式:
分别是通过java实例对象的.getClass()
java类名.class
Class.forName():如果内存中没有指定的字节码文件,这里会有类加载器去寻找、
通过这三种方式找出来的字节码文件都是同一份字节码文件。


public class TestClass {
    public static void main(String[] args) throws Exception {
        String str = "test";
        Class c1 = str.getClass();
        Class c2 = String.class;
        Class c3 = Class.forName("java.lang.String");
        System.out.println(c1== c2);
        System.out.println(c2== c3);
    }
}
System.out.println(int.class == Integer.class);
System.out.println(int.class == Integer.TYPE);

结果是false,true
基本类型的字节码和封装成对象的不是同一份字节码文件。
但是java预定义了基本类型的字节码:即int.class被Integer.TYPE表示。
float.class被Float.TYPE表示....。加上Void.TYPE共有九个。
并且封装了不少的方法用于判断字节码文件是属于那种java类类型的。

java基础整理02--反射的基础知识点_第1张图片
image.png

反射的过程就是把java类中的各种成分映射成相应的java类。

2、Constructor类的基本用法

比如类里面的变量用Field类表示、类里面的方法用Method类表示、构造方法用Constructor类表示...。
通过反射的Constructor类的getInstance()方法获得实例对象:

public class TestClass {
    public static void main(String[] args) throws Exception {
        String str = "test";
        Class c1 = str.getClass();
        Class c2 = String.class;
        Class c3 = Class.forName("java.lang.String");
        System.out.println(c1 == c2);
        Constructor c = c1.getConstructor(StringBuffer.class);//传入参数是指该构造函数实际接收的参数的类型
        String s = c.newInstance(new StringBuffer("abc"));//传入的参数的类型和上面的一致 
        System.out.println(s.charAt(2) + " 通过constructor生成的字符串:" + s);
    }
}

3、Field类

可以通过Class类的GetField来获得Class类里面的所有public的属性(不包含protected和private)。

public class TestClass {
    public static void main(String[] args) throws Exception {
        ReflectPoint r = new ReflectPoint(10, 3);
        Class c = r.getClass();
        Field f = c.getField("x");// 此时f的值不是10,因为f是存在类上的,代表了类上的x属性
        System.out.println(f.get(r));// 通过get方法可以获得r实例对象的x的取值 x是public的
        
        f = c.getDeclaredField("y");//获得的f是代表了类上的y属性(y是私有的)
        f.setAccessible(true);//暴力反射:改变了y的访问方式,变成public的
        System.out.println(f.get(r));
        
         f = c.getDeclaredField("z");// 此时f的值不是0,因为f是存在类上的,代表了类上的z属性
        System.out.println(f.get(r));// 通过get方法可以获得r实例对象的z取值 z是protected
    }

}

4、Method类

得到String类的方法charAt(int index);

Method m = Class.forName("java.lang.String").getMethod("charAt", int.class);
]System.out.println(m.invoke(new String("abc"), 2));

如果invoke()方法的第一个参数设置为null,则调用静态方法。

5、反射调用是数组类型的方法

因为jdk1.5要兼容1.4,这时候,因为1.4没有可变形参,一个数组就相当于多个形参了,那接收数组的一个形参的解决方法该如何呢?
一般就是把数组强制转换成一个Object。
至于为啥要用到反射调用面方法嘛。。。有时候我们运行一个类时,传参数是各种类名,叫我们运行时,只能用到反射啦。

public static void main(String[] args) throws NoSuchMethodException, SecurityException, IllegalAccessException,
            IllegalArgumentException, InvocationTargetException {
        Class testMain = TestMain.class;
        Method method = testMain.getMethod("main", String[].class);
        String[] strs = new String[] { "123", "124", "125" };
        method.invoke(null, new Object[] { strs });
    }

下图的结果可以用上面内容分析:

java基础整理02--反射的基础知识点_第2张图片
image.png

你可能感兴趣的:(java基础整理02--反射的基础知识点)