------- <a href="http://www.itheima.com" target="blank">android培训</a>、<a href="http://www.itheima.com" target="blank">java培训</a>、期待与您交流! ---------
反射:
反射的基石:::Class类
Java程序的各个Java类属于同一类事物,描述这类事物的java类名就是Class;Class类代表Java类,它的各个实例对象分别对应各个类在内存中的字节码.
一个类被类加载器加载到内存中,占用一片内存空间,这个空间里面的内容就是类的字节码,不同的类的字节码是不同的,所以他们在内存中的内容志不同的,这一个个的空间可分别用一个个的对象来表示,这些对象显然具有相同的类型
Java类用于描述一类事物的共性,该类事物有什么属性,没有什么什么属性,至于这个属性的值是什么,则是由这个类的实例对象来确定的,不同的Class,要注意与小写class关键字的区别.Class类描述了那些方面的信息?类的名字,类的访问属性,类所属的包名,字段名称的列表等
得到各个字节码对应的实例对象的三种方式(Class 类型)
类名.class,例如:System.class
对象.getClass(),例如: new Date().getClass();
Class.forName("类名"),例如,Class.forName("java.util.Date");
对象名.isPrimitive();//判断该对象是否是原始数据类型
举例说明:
String str = "abc";
Class cl1 = str.getClass();
Class cl2 = String.class;
Class ci2 = Class.forName("java.lang.String");
cl1==cl2==cl3;
判断是否为数组的原始类型Class.isArray();
int.class==Integer.TYPE;
总之,只要是在源程序中出现的类型,都有各自的Class实例对象
反射:就是把Java类中的各种成分映射成相应的Java类,一个java类中用一个Class类的对象来表示,一个类中的组成部分:成员变量,方法,构造函数,包等等信息也用一个个的java类来表示,就像汽车是一个类,汽车中的发动机,变速箱等等也是一个个的类.表示Java类的Class显然要提供一系列的方法来获得其中的变量,方法,构造方法,修饰符,包等信息,这些信息就是用相应类的实例对象来表示,它们是Field, Method, Constructor,Package等等
一个类中的每个成员都可以用相应的反射API类的一个实例对象来表示,通过调用Class类的方法可以得到这些实例对象.
Constructor类
Constructor类代表某个类中的一个构造方法
得到某个类所有的构造方法:
例子:Constructor[] constructor = Class.forName("java.lang.String").getConstructors();
得到某一个构造方法:
例子:Constructor constructor = Class.forName("java.lang.String").getConstructors(StringBuilder.class);
创建实例对象:
通常方式;String str = new String(new StringBuffer("abc"));
反射方式:String str = (String)constructor.newInstance(new StringBuffer("abc"));
用构造方法创建实例对象:
Constructor constructor1 = String.class.getConstructor(StringBuffer.class);
String str1 = (String)constructor1.newInstance(new StringBuffer("abc"));
char ch = str1.charAt(2);
System.out.println("ch="+ch);//结果为ch=c;
Class.newInstance()方法:直接创建空参数的构造方法(省略中间步骤,获取构造方法类)
Field类:
Field类代表某个类中的一个成员变量
演示用eclips自动生成java类的构造方法
Method类
它代表某个类中的一个成员方法
得到类中某一个方法
例子:Method method= Class.forName("java.lang.String").getMethod("charAt",int.class)(方法名(字符串),传入的参数的字节码)
反射方式:System.out.println(method.invoke(Object obj , 1));
调用方法:
通常方式:System.out.println(str.charAt(1));
如果传递给Method对象的invoke方法的第一个参数为null,说明该Method对象对应的是一个静态方法:
用反射方式执行某个类中的main方法;
目标:写一个程序,这个程序那个根据用户提供的类名,去执行该类中的main方法
数组的反射
具有相同维数和元素类型的数组属于同一类型,即具有相同的Class实例对象.
代表数组的Class实例对象getSuperClass()方法返回的父类的Object类对应的Class
基本类型的一维数组可以被当作Object类型使用,不能当作Object[]类型使用;非基本类型的一维数组,既可以当作Object[]类型使用,又可以当作Object类型使用.
Arrays.asList()方法处理int[]和String[]时的差异,Arrays工具类用于完成对数组的反射操作
int[] a1= new int[3];
int[] a2 = new int[4];
int[][] a3 = new int[2][3];
String[] a4 = new String[3];
System.out,println(a1.getClass() == a2.getClass());//结果为true
System.out,println(a1.getClass() == a4.getClass());//结果为false
System.out,println(a1.getClass() == a3.getClass());//结果为false
System.out,println(a1.getClass().getSuperclass().getName());//结果为java.lang.Object
System.out,println(a4.getClass().getSuperclass().getName());//结果为java.lang.Object
基本数据类型的数组不能转换为Object的数组
Object obj1 = a1;//结果为true
Object obj2 = a4;//结果为true
Object[] obj3 = a1;//结果为false
Object[] obj4 = a4;//结果为true
Object[] obj15= a3;//结果为true
用反射的思想打印基本数据类型的元素
public static void run(Object obj)
{
Class clazz = obj.getClass();
if(clazz.isArray())
{
int len = Array.getLength(obj);
for(int x=0; x<len; x++)
{
System.out.println(Array.get(obj), x);
}
}
else
System.out.println(obj);
}
反射的作用:实现框架功能
框架与框架要解决的核心问题
我做房子卖给用户,由用户自己安装门窗和空调,我做的房子就是框架,用户需要使用我的框架,把门窗插入进我提供的框架中,框架与工具类有区别,工具类被用户的类调用,而框架则是调用用户提供的类:
框架要解决的核心问题
我在写框架的时候,,我怎样保证我写的框架程序怎样能调用到你以后写的类呢?
因为在写程序是无法知道要被调用的类名,所以,在程序中无法直接new某个类的实例对象了,而要用到反射方式来做