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

来源:互联网 发布:java gbk转utf8 黑马 编辑:程序博客网 时间:2024/04/29 04:32

-----------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”});编译器会作特殊处理,编译时不把参数当作数组看待,也就不会数组打散成若干参数了.

 

        

0 0