您的位置:首页 > 职场人生

黑马程序员--java反射学习笔记

2015-06-22 19:35 387 查看
-----------android培训java培训、java学习型技术博客、期待与您交流!------------

一. 反射的基础.

a) 反射的基石:Class类

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

ii. 对此提问:众多的人用一个什么类表示:Person

iii. 对此提问:Person类代表人,它的实例对象就是张三,李四这样一个个具体的对象,Class类代表Java类,它的各个实例对象又分别对应什么了

1. 对应各个类在内存中的字节码,例如Person类的字节码,ArrayList类的字节码.

2. 一个类被类加载器加载到内存中,占用一片存储空间,这个空间里面的内容就是类的字节码,不同的类的字节码是不同的,所以它们在内存中的内容是不同的,这一个个的空间可分别用一个个的对象来表示,这些对象显然具有相同的类型,这个类型是什么了

iv. 如何得到各个字节码对应的实例对象(Class类型)

1. 类名.class;如:System.class

2. 对象.getClass();如:new Date().getName();

3. Class.forName(“类名”)如Class.forName(“java.util.Date”);

v. 九个预定义Class实例对象:

1. 参看Class.isPrimitive方法的帮助

2. Int.class==Integer.TYPE

3. 8个基本的类型对应了八个Class对象,void也对应了一个Class对象

4. 数组也是一种类型:数组类型的Class实例对象

a) Class isArray()

b) 总之,只要是源程序中出现的类型,都有各自的Class实例对象,例如:int][];void;

b) 反射就是把Java类中的各种成分映射成相应的Java类,例如,一个java类中用一个Class类的对象来表示,一个类中的组成部分:成员变量,方法,构造方法,包等等信息也用一个个的JAVA类来表示,就像汽车是一个类,汽车中的发动机,变速箱等等也是一个个的类;表示java类的Class类显然要提供一系列的方法,来获得其中的变量,方法,构造方法,修饰符,包等信息,这些信息就是用相应类的实例对象来表示,它们是Field,Method,Contructor,Package等等.

c) 一个类中的每个成员都可以用相应的反射API类的一个实例对象来表示,通过调用Class类的方法可以得到这些实例对象后,得到这些实例对象后有什么用呢,怎么用了,这正是学习和应用反射的要点.

二. Constructor类

a) Constructor类代表某个类中的一个构造方法

b) 得到某个类所有的构造方法:

i. 例子:Constructor[]constructor = Class.forName (“java.lang.String).getConstructor(StringBuffer.class);

ii. 程序开发分为编译时和开发时.编译器只看变量的定义,不看代码的执行.

iii. 得到某一个构造方法:

1. Constructor constructor =Class.forName(“java.lang.String”).getConstruntor(StringBuffer(“abc”));//调用获得的方法时要用到上面相同类型的实例对象.

iv. 创建实例对象:

1. 通常方式:String str= new String(new StringBuffer(“abc”));

2. 反射方法:String str= (String)constructor.newinstance(new StringBuffer(“abc”));//调用获得的方法时要用到上面相同类型的实例对象.

v. Class.newInstance()方法:

1. 例子:String obj =(String)Class.forName(“java.lang.String”).newInstance();

2. 该方法内部先得到默认的构造方法,然后用该构造方法创建实例对象.

3. 该方法内部的具体代码是怎么样写的了,用到了缓存机制来保存默认构造方法的实例对象.

三. Field类

a) Field类代表某介盆地辊的一个成员变量,

b) 演示用eclipse自动生成java类的构造方法.

c) 问题:得到的Field对象是对应到类上面的成员变量,还是对应到对象上的成员变量;类只有一个,而该类的实例对象有多个,如果是与对象关联,那关联的是哪个对象了,所以字段fieldX代表的是X的定义,而不是具体的x(具体的对象)变量

d) 示例代码

ReflectPoint pt1 = new ReflectPoint(3,5);

Field fieldy = pt1.getClass().getField(“y”);

e) 功能需求:将任间一个对象中的所有String类型的成员变量所对应的字符串内容中的b改成a

package com.jwd.black.ReflectPoint;

import java.lang.reflect.Field;

publicclass ReflectTest {
publicstaticvoid main(String[]args)
throws Exception{
ReflectPointpt1 =newReflectPoint(3,5);
Field fieldy =pt1.getClass().getField("y");

changeString(pt1);
System.out.print(pt1);
}
privatestaticvoid changeString(Objectobj)
throws Exception{
Field[]
fields = obj.getClass().getFields();
for(Fieldfield:fields){
if(field.getType()==String.class){
String
oldValue = (String)field.get(obj);
String
newValue = oldValue.replace('b','a');
field.set(obj,newValue);
}
}
}
}

package com.jwd.black.ReflectPoint;

publicclass ReflectPoint {
privateintx;
publicinty;
public Stringstr1 =
"ball";
public Stringstr2 =
"basketball";
public Stringstr3 =
"itcast";

ReflectPoint(intx,inty)
{
this.x =x;
this.y =y;
}

public String toString() {
returnstr1 +";" +
str2 +
";" +str3 +
";";
}
}

f) 总结:获得一个对象,换掉里面字段的内容.

四. Method类:成员方法的反射

a) Method类代表某个类中的一个成员方法

b) 得到类中的某一个方法

i. 例子:Method charAt= Class.forName(“java.lang.String”).getMethod(“charAt”,int.class);

ii. 调用方法

1. 通常方式:System.out.printIn(str.charAt(1));

2. 反射方式:System.out.printIn(charAt.invoke(str,1);

3. 如果传递给Method对象的invoke()方法的一个参数为null,这有着什么样的意义了,说明该Method对象对应的是一个静态方法

c) 对接收数组参数的成员方法进行反射,用反射方式执行某个类中的main()方法

i. 目标:写一个程序,这个程序能够根据用户提供的类名,去执行该类中的main方法

ii. 问题:启动Java程序的main方法的参数是一个字符串数组,即public static voidmain(String[] args),通过反射方式来调用这个main方法时,如何为invoke方法传递参数了à在给main方法传递参数时,不能使用代码mainMethod.invoke(null,new String[]{“xxx”};

iii. 解决办法:

mainMethod.invoke(null,new Object[]{new String[]{“xxx”}});

mainMethod.invoke(null,( Object)newString[]{“xxx”});编译器会作特殊处理,编译时不把参数当作数组看待,也就不会数组打散成若干参数了.
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: