黑马程序员技术交流社区

标题: 反射小结 [打印本页]

作者: wumuzhong    时间: 2013-9-30 22:32
标题: 反射小结
java程序中的各个java类属于同一类事务,描述这类事务的java类名就是Class
对比提问:众多的人用一个什么类表示?众多的java类用一个什么类表示?
  人-->Person
  java类-->Class
对比提问:Person类代表人,它的实例对象就是张三,李四这样一个个具体的人,Class类代表java类,它的各个实例对象又分别对应什么呢?
  -> 对应各个类在内存中的字节码,例如,Person类的字节码,ArrayList类的字节码,等等
  ->一个类被类加载器加载到内存中,占用一片存储空间,这个空间里面的内容就是类的字节码,不同的类的字节码是不同的,所以它们在内存中的内容是不同的,这一个个  的空间可分别用一个个的对象了表示,这些对象显然具有相同的类型,这个类型是什么呢?
如何得到各个字节码对应的实例对象(Class类型)
  -》类名.class  例如:System.class
  -》对象.getClass() 例如:new Date().getClass()
  -》Class.forName("类名")例如:Class.forName("Java.util.String);  
九个预定义Class实例对象
-》参看Class.isPrimitive方法的帮助 是否是基本数据类型
-》int.class ==Integer.TYPE
数组类的Class实例对象
-》Class.isArray()
-》总之,只要是在源程序中出现的类型,都有各自的Class实例对象
public class Test{
public static void main(String args[]){
String str1 = "abc";
Class cls1 = str.getClass();
Class cls2 = String.class;
Class cls3 = Class.forName("java.lang.Strign");
System.out.println(cls1 == cls2);
System.out.println(cls2 == cls3);  
System.out.println(cls1.isPrimitive());
System.out.println(int.class.isPrimitive());
System.out.println(int.class == Integer.class);
System.out.println(int.class == Integer.TYPE);
System.out.println(int[].class.isPrimitive());
}
}
  反射就是把java类中的个各成分映射成相应的java类,
一个类中的每个成员都可以用相应的反射API类的一个实例对象来表示,
Constructor类
Constructor类代表某个类中的一个构造方法
得到某个类所有的构造方法:
-》例子:Constructor[] constructor = Class.forName("java.lang.String").getConstructor();
得到某一个构造方法
-》例子:Construct constructor = Class.forName("java.lang.String").getConstructor(StringBuffer.class);
创建实例对象:
-》通常方式:String str = new String(new StringBuffer("abc"));
-》反射方式:String str = (String)constructor.newInstance(new StringBuffer("abc"));
Class.newInstance()方法
-》例子:String obj = (String)Class.forName("java.lang.String").newInstance();
-》该方法内部先得到默认的构造方法,然后用该构造方法创建实例对象
-》该方法内部的具体代码是怎样写的呢?用到了缓存机制来保存默认构造方法的实例对象
Field类
Field类代表某个类中的一个成员变量
问题:得到的Field对象对应到类上面的成员变量,还是对应到对象上的成员变量?类只有一个,而该类的实例对象有多个,如果是与对象关联,那关联的是那个对象呢?所以字段fieldX代表的是x的定义,而不是具体的x变量
私有的变量要想访问必须调用setAccessible(true)才可以访问
示例代码
Point point = new Point(1,7);
Field y = Class.forName("cn.corejava.Point").getField("y");
System.out.println(y.get(point));
Field x = Class.forName("cn.corejava.Point").getField("x");
x.setAccessible(true);
System.out.println(x.get(point));
将任意一个对象中的所有Strign类型的成员变量所对应的字符串中的”b”改成“a"
private static void changeStringValue(Object obj) throws Exception{
Field[] fields = obj.getClass().getFields();
for(Field field:fields){
  if(field.getType() == String.class){
  String oldValue = (String)field.get(obj);
  String newValue = oldValue.replace('b','a');
  field.set(obj,newValue);
}
}

Method类
Method类代表某个类中的一个成员方法
得到类中某一个方法:
例子:  Method charAt = Class.forName("java.lang.String").getMethod("charAt",int.class);
调用方法
通常方式:System.out.println(str.charAt(1));
反射方式:System.out.pritnln((charAt.invoke(str,1));
  如果传递给Method对象的Invoke()方法的一个参数为null,这有什么意义呢?说明该Method对象对应的是一个静态方法
JDK1.4和JDK1.5的Invoke方法的区别
JDK1.5:public Object invoke(Object obj,Object ...args);
JDK1.4:public Object invoke(Object obj,Object[]args),即按jdk1.4的语法需要将一个数组作为参数传递给invoke方法是,数组中的每个元素分别对应被调用方法中的一个参数

作者: 蔚蓝天色    时间: 2013-10-1 00:02
得到某个类所有的构造方法:
-》例子:Constructor[] constructor = Class.forName("java.lang.String").getConstructor();

纠正一下:获得所有的构造方法貌似是:getConstructors()
作者: wumuzhong    时间: 2013-10-1 09:26
嗯嗯 手误 谢谢纠正
作者: 漫步人    时间: 2013-10-23 12:34
今天复习了一下张老师的java基础加强_反射(总结)




欢迎光临 黑马程序员技术交流社区 (http://bbs.itheima.com/) 黑马程序员IT技术论坛 X3.2