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类中的各种成分映射成相应的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<String> 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 = TestMain.class;
Method method = testMain.getMethod("main", String[].class);
String[] strs = new String[] { "123", "124", "125" };
method.invoke(null, new Object[] { strs });
}
下图的结果可以用上面内容分析: