黑马程序员-高新技术(反射)

-----------android培训java培训、java学习型技术博客、期待与您交流! ------------


一、反射的基石--Class类


1、定义:

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


2、对比提问1:


众多的人用一个什么类表示?众多的java类用一个什么类表示?
    人类---Person
    java类--Class
    注意这里Class是大写的,不是关键字class。

3、对比提问2:


(1)Person类代表人,它的实例对象就是张三,李四这样一个个具体的人,
(2)Class类代表java类,它的各个实现对象又分别对应什么呢?
答:对应各个类在内存中的字节码,例如Person类的字节码,ArrayList类的字节码,等等。
一个类被类加载器加载到内存中,占用一片存储空间,这个空间里面的内容就是类的字节码,不同的类的字节码是不同的,所以它们在内存中的内容是不同的,这一个个的空间分别用一个个的对象来表示,这些对象显然具有相同的类型,这个类型是什么呢?就是Class类。

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


(1)类名.class,例如System.class
(2)对象.getClass(),例如new Date().getClass()
(3)Class.forName("类名"),例如 Class.forName("java.util.Data");

获取这个Class对象,有三种方式:
1:通过每个对象都具备的方法getClass来获取。弊端:必须要创建该类对象,才可以调用getClass方法。
2:每一个数据类型(基本数据类型和引用数据类型)都有一个静态的属性class。弊端:必须要先明确该类。
 前两种方式不利于程序的扩展,因为都需要在程序使用具体的类来完成。
3:使用的Class类中的方法,静态的forName方法。
 指定什么类名,就获取什么类字节码文件对象,这种方式的扩展性最强,只要将类名的字符串传入即可。
// 1. 根据给定的类名来获得  用于类加载
String classname = "cn.itcast.reflect.Person";// 来自配置文件
Class clazz = Class.forName(classname);// 此对象代表Person.class
// 2. 如果拿到了对象,不知道是什么类型   用于获得对象的类型
Object obj = new Person();
Class clazz1 = obj.getClass();// 获得对象具体的类型
// 3. 如果是明确地获得某个类的Class对象  主要用于传参
Class clazz2 = Person.class;


5、九个预定义Class实例对象:

八个基本数据类型(boolean,char,byte,short,int,lang,float,double)
八个基本数据类型外加void
参看 Class.isPrimitive方法的帮助

int.class == Integer.TYPE

6、数组类型的Class实例对象


    Class.isArray()

7、总之,只要是在源程序中出现的类型,都有各种的Class实例对象。


    例如int[],void。

8、代码示例:

class Demo{
    public static void main(String[] args) throws Exception{
        String str1 = "abc";
        Class cls1 = str1.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 == Integer.class); //false
        System.out.println(int.class == Integer.TYPE);  //true
    }   
}

二、反射的概念


反射就是把java类中的各种成分映射成相应的java类。
也可以理解成反射就是程序自己能够检测自身信息,就像人会通过镜子来查看自己的身体。

例如,一个java类中用一个Class类的对象来表示,一个类中的组成部分:成员变量,方法,构造方法,包等等信息也是用一个个java类来表示。

就像骑车是一个类,骑车中的发动机,变速箱等等也是一个个类,表示java类的Class类显然要提供一系列的方法,来获得其中的变量,方法,构造方法,修饰符,包等信息,这些信息就是用相应类的实例对象来表示,他们是Field、Method、Contructor、Package等等。

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

 

三、JAVA反射的常规使用步骤

 

反射调用一般分为4个步骤:

1、得到要调用类的class对象

2、对该class对象调用newInstance()方法,得到要调用类的实例对象

3、得到要调用的类中的方法(Method)

4、方法调用(invoke)

四、构造方法的反射


Constructor 类

Constructor类代表某个类中的一个构造方法。
得到某个类所有的构造方法:
例子: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 obj = (String)Class.forName("java.lang.String").newInstance();

该方法内部先得到默认的构造方法,然后用该构造方法创建实例对象。
该方法内部的具体代码是怎么样写的呢?用到了缓冲机制来保存默认构造方法的实例对象。

一个类有多个构造方法,用什么方式可以区分清楚想要得到其中的哪个方法呢?根据参数的个数和类型,
例如,Class.getMethod(name.Class...args)中的args参数就代表索要获取的那个方法的各个参数的类型的列表。
重点:参数类型用什么方式表示?用Class实例对象。
例如:
int.class,(int []).class
int [] ints = new int[0];
ints.getClass();

代码示例:

class Demo{
    public static void sop(Object obj){System.out.println(obj);}
	public static void main(String[] args) throws Exception {
		String s1 = "1234";
		Class c1 = s1.getClass();
		Class c2 = String.class;
		Class c3 = Class.forName("java.lang.String");
		sop(c1==c2);                    //c1与c2是否是同一个对象true
		sop(c1==c3);                    //c1与c3是否是同一个对象true
		sop(String.class.isPrimitive());//String是否是基本类型false
		sop(int.class.isPrimitive());   //int是否是基本类型true
		sop(int.class==Integer.class);  //int与Integer的字节码是否是同一个对象false
		sop(int.class==Integer.TYPE);   //int与Integer.TYPE的字节码是否是同一个对象true
		sop(int[].class.isPrimitive()); //int[]是否是基本类型false
		sop(int[].class.isArray());     //int[]是否是数组类型true
	}
}

五、成员变量的反射


Field类
Field类代表某个类中的一个成员变量,
Field可以用eclipse自动生成Java类的构造方法
问题:得到的Field对象是对应到类上面的成员变量,还是对应到对象上的成员变量?
类只有一个,而该类的实例对象有多个,如果是与对象关联,那么关联的是哪个对象呢?
所以字段fieldX代表的是x的定义,而不是具体的x变量。

代码示例:


Point类的构造函数

public Point(int x,int y)在坐标空间中指定的(x,y)位置构造并初始化一个点。

class Point {
	private int x;
	public int y;
	public String s1="abababab";
	public String s2="aaaabbbb";
	
	public Point(int x, int y) {
		super();
		this.x = x;
		this.y = y;
	}
	public Point(int x, int y, String s1, String s2) {
		super();
		this.x = x;
		this.y = y;
		this.s1 = s1;
		this.s2 = s2;
	}

	//toString()返回此点的字符串表示形式及其在 (x,y) 坐标空间中的位置。
	public String toString(){
		return s1+"---"+s2;
	}
}

下面的这个单独的程序实现对上面这个类的成员变量的反射。

import java.lang.reflect.Field;
class FieldRef{
	public static void main(String[] args) throws Exception{
		//创建一个坐标
		Point point = new Point(5, 8);
		
		//获取字段y的值
		Field fieldY = point.getClass().getField("y");
		
		//获取public修饰的字段
		System.out.println("输出public属性字段"+fieldY.get(point));
		
		//获取字段的值
		Field fieldX = point.getClass().getDeclaredField("x");
		
		//获取类中的类字段
		fieldX.setAccessible(true);
		
		//获取private修饰的字段
		System.out.println("输出private属性字段"+fieldX.get(point));
		
		//调用chang方法
		chang(point);
		System.out.println(point);
	}
	//通过反射改变字段中的字母
	public static void chang(Object obj) throws Exception{
		
		//创建字段数组,获取所有成员字段
		Field[] fields = obj.getClass().getFields();
		for(Field field : fields){
			
			//如果类型为String类型
			if(String.class == field.getType()) {
				String oldValue = (String) field.get(obj);
				
				//获取成员字段值,替换该值
				String newValue = oldValue.replace('a', 'b');
				//设置成员字段的值
				field.set(obj, newValue);
			}
		}
	}
}

六、数组的反射


具有相同维数和元素类型的数组属于同一个类型,即具有相同的Class实例对象。
代表数组的Class实例对象的getSuperClass()方法返回的父类为Object类对应的Class。
基本类型的一维数组可以被当作Object类型使用,不能当作Object[]类型使用,
非基本类型的一维数组,既可以当做Object类型使用,又可以当做Object[]类型使用。
Arrays.asList()方法处理int[]和String[]时的差异。
Array工具类用于完成对数组的反射操作。

import java.lang.reflect.Array;
import java.util.Arrays;

class ArrayRef {
	public static void sop(Object obj){System.out.println(obj);}
	public static void main(String[] args){
		//定义几个不同维数的数组
		int[] a1 = new int[]{1,2,3};
		int[] a2 = new int[4];
		int[][] a3 = new int[2][3];
		String[] a4 = new String[]{"1","2","3"};
		
		sop(a1.getClass()==a2.getClass());			//a1与a2的字节码是否相同
		sop(a3.getClass().getName());				//获取a3数组的名字
		sop(a1.getClass().getSuperclass().getName());//a1数组父类的名字
		
		sop("------------------------");
		Object obj1 = a1;
		//Object[] obj2=a1;//两个对象类型不同,不能运算
		Object boj3 = a4;
		Object[] boj4 = a4;
		Object boj5 = a3;
		Object[] boj6 = a3;
		sop("无工具类Arrays的输出:"+a1);
		sop("无工具类Arrays的输出:"+a4);
		sop("有工具类Arrays的输出:"+Arrays.asList(a1));
		sop("有工具类Arrays的输出:"+Arrays.asList(a4));
		printObject(a1);	
		printObject(1);
	}
	//打印对象中成员方法
	private static void printObject(Object obj) {
		Class cla = obj.getClass();
		//如果是是一个数组,就打印该数组中所有对象
		if(cla.isArray()){
			int len = Array.getLength(obj);			
			sop("\n调用自定义方法的数组的输出");
			for (int i = 0; i < len; i++) {
				sop(Array.get(obj, i));
			}
		}
		else{
			sop("调用自定义方法的普通对象的输出\n"+obj);
		}
	}
}

七、集合的反射

 

思路:
1、通过文件输出流,创建一个文件Config.properties,
2、获取”className=java.util.ArrayList“的字节码后存入文件中。文件可以手动创建。
3、创建文件输入流和一个属性集,读取指定文件内容,加载到属性集中。
4、用键”className“在属性集中收索。
5、获得新集合,往集合中添加元素。
6、打印集合

代码示例:

import java.io.*;
import java.util.*;

class CollectRef {
	public static void main(String[] args) throws Exception {
		//在当前目录下创建一个文件,并写入指定数据,可以手动创建该文件。
		OutputStream os = new FileOutputStream("Config.properties");
		os.write("className=java.util.ArrayList".getBytes());
		os.close();
		
		//创建文件输入流对象,读取数据
		InputStream is = new FileInputStream("Config.properties");
		//创建属性集
		Properties props = new Properties();
		//加载输入流对象
		props.load(is);                     
		is.close();
		
		//用指定的键在此属性集中搜索属性
		String className = props.getProperty("className");

		//创建此Class对象所表示的类的一个新集合
		Collection cols = (Collection)Class.forName(className).newInstance();
		cols.add("1");
		cols.add("2");
		cols.add("5");
		cols.add("3");
		System.out.println("集合中的元素:"+cols);
	}
}

 

 

八、通过反射获得泛型的实际类型参数


 分析:
 比如:Vector<Date> v = new Vector<Date>();
 那么通过v是无法知道定义它的那个泛型类型的,那么可以把这个v交给一个方法当做参数或者返回值类型来使用,
 然后通过Method类的getGenericParameterTypes()方法来获得该方法的参数列表,从而获得参数实际间类型。

代码示例:
import java.lang.reflect.Method;
import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;
import java.sql.Date;
import java.util.Vector;

/**
 * 通过反射获得泛型的实际类型参数
 * @author Administrator
 */
public class GenericTest {
	public static void main(String[] args) throws Exception {
		//通过v对象是无法得到的,值能通过方法来获取,那么就要定义一个方法如下面applyVector方法。
		//Vector<Date> v = new Vector<Date>();
		
		//获得字节码,通过字节码获得方法,参数是一个方法名,Vector的字节码
		Method applyMethod = GenericTest.class.getMethod("applyVector", Vector.class);
		
		//通过Method类的getGenericParameterTypes()方法
		//反射applyMethod方法的参数化类型,可能有多个,所以是数组。
		Type[] types = applyMethod.getGenericParameterTypes();
		
		//返回types的第一个参数,返回的是ParameterizedType类型。
		ParameterizedType pType = (ParameterizedType)types[0];
		
		//获得原始类型
		System.out.println(pType.getRawType());
		
		//获得实际类型。
		System.out.println(pType.getActualTypeArguments()[0]);
	}
	
	//需要定义这个方法,通过这个方法的参数列表来反射他的参数类型。
	public static void applyVector(Vector<Date> v) {
	}
}
/*
结果
class java.util.Vector
class java.sql.Date
*/

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值