反射

一、反射简介

jdk1.2开始出现

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

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

java反射机制是在运行状态中,对于任意一个类(class文件),都能知道这个类的所有属性和方法;

对于任意一个对象,我们都能够调用它的任意一个方法和属性;

这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制。

可以理解为解剖。

要想对字节码文件进行解剖,必须要有字节码文件对象。

 

获得字节码文件的三种方法

1Object类中的getClass()方法。

   想要用这种方式,必须要明确具体的类,并创建对象,麻烦;

2、任何数据类型都具备一个静态属性.class获取其对应的Class对象。

    相对简单,但是还是要明确用到类中的静态成员。还是不够扩展。

3、只要通过给定的类的字符串名称就可以获取该类,更为扩展,

   可用Class类中的方法完成,该方法是forName

    

  new时,现根据被new的类的名称找寻该类的字节码文件,并对加载进内存,并创建该字节码文件对象,并接着创建该字节码文件对应的类的对象

  

  九个预定义Class实例对象,表示八个基本类型和 void。这些类对象由 Java 虚拟机创建,与其表示的基本类型同名,即 booleanbytecharshortintlongfloat 和 double。 获得方法:int.class......

  使用

 

  基本包装类的获取类的基本类型的字节码的方法

  Boolean.TYPE, Character.TYPE, Byte.TYPE, Short.TYPE, Integer.TYPE, Long.TYPE, Float.TYPE, Double.TYPE, Void.TYPE

 

  数组类型的Class实例对象

  Class.isArray() 判断是不是数组

 

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

 

 

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

   得到某个类所有的构造方法getConstructor(),如:

     Constructor[] constructor = Class.forName("java.lang.String").getConstructor()

   得到某一个构造方法getConstructor(字节码),如:

     Constructor 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 str = (String)Class.forName("java.lang.String").newInstance();

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

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

  

Field类代表某个类中的成员变量

    get(obj)获得obj这个对象中指定的成员变量的值

    获得Field的方法,如

    Field fieldx = rp.getClass().getField("x");//获取公有的车成员变量

   

    Field fieldy = rp.getClass().getDeclaredField("y"); //获取私有的成员变量   

    fieldy.setAccessible(true);//暴力获取

 

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

    得到一个类中的某个方法:

     Method charAt = Class.forName("java.lang.String").getMethod("charAt".class);

    调用方法

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

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

        如果传递给Method对象的invoke()方法的第一个参数nullMethod对象对应的是一个静态方法

 

二、实例演示

 1、实例一

package Javaenhance;

/**

 * 时间:2014/6/9 星期一 22:44:20

 * @author qiwenming

 * @说明 演示反射中,获取字节码的三种方式

 *      ①类名.class

 *      ②对象.getClass()

 *      Class.forName()

 *     和一些与相关的方法

 */

public class Javaenhance_3_reflectDemo01 {

public static void main(String[] args) throws ClassNotFoundException {

 

        //①类名.class

Class cls0 = String.class;

//②对象.getClass()

String str = "qwm";

Class cls1 = str.getClass();

//Class.forName()

Class cls2 = Class.forName("java.lang.String");

//使用的是同一个字节码文件

System.out.println(cls0==cls1);

System.out.println(cls0==cls2);

//isPrimitive 判读指定的类型是不是基本类型

System.out.println("是基本数据类型吗?"+cls0.isPrimitive());

System.out.println("是基本数据类型吗?"+int.class.isPrimitive());

System.out.println("是基本数据类型吗?"+int[].class.isPrimitive());

//TYPE表示基本类型的 Class 对象。

System.out.println(int.class==Integer.TYPE);

//判读字节码是不是数组

System.out.println(int[].class.isArray());

}

}

 

 2、实例二

package Javaenhance;

import java.lang.reflect.Constructor;

import java.lang.reflect.InvocationTargetException;

/**

 * 时间:2014/6/10 星期二 0:04:24

 * @author qiwenming

 * @说明 这里是演示使用构造方法创建一个对象

 *    步骤:

 *      ①通过字节码创建获得构造函数。

 *      ②通过构造函数创建对象

 */

 

public class Javaenhance_3_reflectDemo02_constructor {

 

public static void main(String[] args) throws NoSuchMethodException, SecurityException, InstantiationException, IllegalAccessException, IllegalArgumentException, InvocationTargetException {

 

//使用反射获得参数为StringBuilder的构造函数

Constructor constructor = String.class.getConstructor(StringBuilder.class);

    //使用构造方法创建一个对象,返回的是Object,这里的错误时编译时的错误

String str = (String)constructor.newInstance(new StringBuilder("nihaoma"));

System.out.println(str.toCharArray()[2]);

}

}

 

 3、实例三

package Javaenhance;

import java.lang.reflect.Field;

/**

 * 时间;2014/6/10 星期二 0:45:59

 * @author qiwenming

 * @说明 使用反射获取成员变量,边修改类中成员变量的值

 */

public class Javaenhance_3_reflectDemo03_membervariable {

 

public static void main(String[] args) throws Exception {

//创建一个对象

Reflectpointer rp = new Reflectpointer(4, 6);

//使用反射获得公共成员变量x,拿到的x不是对象上的上x,而是类上的x,通过getField拿到的是公有的

Field fieldx = rp.getClass().getField("x");

//打印成员变量的值   想要获取通过使用反射获取的成员变量的值,使用该get方法

        System.out.println(fieldx.get(rp));

        //下面是获取 ,私有的成员变量

Field fieldy = rp.getClass().getDeclaredField("y");

//暴力获取

fieldy.setAccessible(true);

System.out.println(fieldy.get(rp));

System.out.println(rp.toString());

//把字节码中,类型为String的成员变量值中含有的'i'替换为'q'

changefieldvalue(rp);

System.out.println(rp.toString());

 

}

 

private static void changefieldvalue(Object obj) throws Exception {

//获取所有的成员变量

Field fields[] = obj.getClass().getFields();

//筛选出类型为String的成员变量,并按要求修改值

for (Field field : fields) { 

//因为是同一份字节码所以这里使用"=="更合适

if(field.getType() == String.class){

String oldstr = (String)field.get(obj);

String newstr = oldstr.replace('i', 'q');

field.set(obj, newstr);//设置值

}

}

}

 

}

 

 

 4、实例四

package Javaenhance;

import java.lang.reflect.Method;

/**

 * 时间:2014/6/10 星期二 12:43:22

 * @author qiwenming

 * @说明 运用反射,获得成员方法,并执行方法

 */

public class Javaenhance_3_reflectDemo04_membermethod {

 

public static void main(String[] args) throws Exception {

//获得String类中charAt方法 int.class 代表这个方法需要传递一个参数

Method charAt = Class.forName("java.lang.String").getMethod("charAt",int.class);

        //调用方法

System.out.println(charAt.invoke(new String("nihao"), 2));

System.out.println(charAt.invoke(new String("nihao"), 4));

}

}

 

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值