Java学习笔记——反射

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

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

Class.forName的作用:返回类的字节码,返回的方式有两种,一种是这个类曾被加载过,则直接从JVM中返回字节码,另一种是JVM中没用这个类的字节码,得用类加载器先将字节码加载到JVM中,然后返回字节码。

得到类的字节码的三种方式
1.类名.class
2.对象.getClass();
3.Class.forName("完整类名");例如: Class.forName("java.lang.String");(多用于反射)

public class ReflectTest {

    public static void main(String[] args) throws Exception{
        
        String str = "abc";
        Class cls1 = str.getClass();
        Class cls2 = String.class;
        Class cls3 = Class.forName("java.lang.String");
        
        System.out.println(cls1==cls2);//true
        System.out.println(cls1==cls3);//true
        
        //判断是否是基本类型
        System.out.println(cls1.isPrimitive());//false
        System.out.println(int.class.isPrimitive());//true
        System.out.println(int[].class.isPrimitive());//false
        
        System.out.println(int.class==Integer.class);//false
        System.out.println(int.class==Integer.TYPE);//true
        
    }

}
View Code
//用反射实现    new String(new StringBuffer("abc")
    Constructor constructor =     String.class.getConstructor(StringBuffer.class);
    String str1 = (String) constructor.newInstance(new StringBuffer("abc"));
    System.out.println(str1);//abc
View Code

 

成员变量的反射(Field类)

//定义一个Point类
public class Point {
    
    private int x;
    public int y;
    
    public Point(int x, int y) {
        super();
        this.x = x;
        this.y = y;
    }

}

用反射映射Point类

Point p = new Point(3,4);
Field filedY = p.getClass().getField("y");//getField只能得到可见的属性
System.out.println(filedY.get(p));//得到对象p在属性y上的具体值
   
Field filedX = p.getClass().getDeclaredField("x");//getDeclaredField可以得到私有属性
filedX.setAccessible(true);//暴力反射,将私有的设置为可访问的
System.out.println(filedX.get(p));

成员变量反射练习
public class Cat {
    
    String name = "Happy";
    String eyes = "Black";
    String eat = "beef";
    @Override
    public String toString() {
        return "Cat [name=" + name + ", eyes=" + eyes + ", eat=" + eat + "]";
    }
    
}
public class ReflectTest2 {

    public static void main(String[] args) throws Exception {
        
        Cat cat = new Cat();
        System.out.println(cat);//Cat [name=Happy, eyes=Black, eat=beef]
        ChangeCat(cat);
        System.out.println(cat);//Cat [name=Hbppy, eyes=Blbck, eat=beef]

    }

    private static void ChangeCat(Cat obj) throws Exception {
        
        Field[] fields = obj.getClass().getDeclaredFields();
        
        for(Field field:fields){
            field.setAccessible(true);
            if(field.getType()==String.class){
                String oldStr = (String) field.get(obj);
                String newStr = oldStr.replace('a', 'b');
                field.set(obj, newStr);
            }
        }
        
    }

}
View Code

成员方法反射(Method类)

public class ReflectTest2 {

    public static void main(String[] args) throws Exception {
        
        String str1 = "abc";
        Method method =  Class.forName("java.lang.String").
            getMethod("charAt", int.class);
        char ch = (char) method.invoke(str1, 1);
        System.out.println(method.invoke(str1, 1));//b
        System.out.println(method.invoke(str1, new Object[]{2}));//c

    }

}
View Code

对含有数组参数的成员方法进行反射

//定义一个含有main方法的类
class TestArguments{
    public static void main(String[] args){
        
        for(String arg : args){
            System.out.println(arg);
        }
    }
}

public class ReflectTest {

    public static void main(String[] args) throws Exception {
        
        
        /*用反射来实现调用main方法*/
        String className = args[0];
        Method mainMethod = Class.forName(className).getMethod("main", String[].class);
        mainMethod.invoke(null, (Object)new String[]{"111","222","333"});
        mainMethod.invoke(null, new Object[]{new String[]{"111","222","333"}});

       //以上两种调用等价,jdk1.5后,会将数组参数进行拆解,这样上述的一个数组参数就变成了3个参数,则会出现数组个数不匹配,
      //故应将数组强制转换成Object或将数组作为一个对象用Object数组进行封装

    }

}
View Code

 

posted @ 2015-03-14 19:32  N_ll  阅读(124)  评论(0编辑  收藏  举报