目录java
java基础之反射数组
当程序主动使用某个类时,若是该类尚未被加载到内存中,则系统会经过加载、链接、初始化这三个步骤对该类进行初始化。有时会把这一整个流程统称为类加载或类初始化。
类加载指的是将类的class文件读入内存中,并为之建立一个 java.lang.Class 对象,也就是说程序使用任何类的时候,都会为其建立一个class对象。ide
类被加载以后,系统会为之生成一个Class对象,接着会进入链接阶段,链接阶段负责把类的二进制数据合并到JRE中。类的链接又分为下面三个阶段:spa
在java中对类变量指定初始值得方法有两种:1. 声明类变量时指定初始值;2. 使用静态初始化块为类变量指定初始值。翻译
类加载器负责将.class文件加载到内存中,并为之生成对应的Class对象。类加载器负责加载全部的类,系统为全部加载到内存中的类生成一个java.lang.Class 的实例。设计
类加载器的组成:代理
Java反射就是在运行状态中,对于任意一个类,都可以知道这个类的全部属性和方法;对于任意一个对象,都可以调用它的任意方法和属性;而且能改变它的属性。
反射机制容许程序在运行时取得任何一个已知名称的class的内部信息,包括包括其modifiers(修饰符),fields(属性),methods(方法)等,并可于运行时改变fields内容或调用methods。那么咱们即可以更灵活的编写代码,代码能够在运行时装配,无需在组件之间进行源代码连接,下降代码的耦合度;还有动态代理的实现等等。code
java程序中许多对象在运行时会出现两种类型:运行时类型和编译时类型,例如Person p = new Student();这句代码中p在编译时类型为Person,运行时类型为Student。程序须要在运行时发现对象和类的真实信心。而经过使用反射程序就能判断出该对象和类属于哪些类。对象
Java文件被编译后,生成了.class文件,JVM此时就要去解读.class文件。当程序主动去使用某个类时,JVM会经过前面提到的三个步骤:加载、链接和初始化三个步骤对类进行初始化。被编译后的Java文件.class也被JVM解析为一个对象,这个对象就是java.lang.Class。这样当程序在运行时,每一个java文件就最终变成了Class类对象的一个实例。咱们经过Java的反射机制应用到这个实例,就能够去得到甚至去添加改变这个类的属性和动做,使得这个类成为一个动态的类。
Class类的概念尽管很抽象,可是无疑,它是反射机制的起源,是Java语言中一个精巧美妙地设计。
下面是翻译后的中文文档的描述:
Class类的实例表示正在运行的Java应用程序的类和接口。枚举是一种类,注释(注解)是一种接口。每一个数组属于被映射为Class对象的一个类,全部具备相同元素类型和维数的数组都共享该Class对象。基本的Java类型(boolean、byte、char、short、int、long、float 和 double)和关键字 void 也表示为 Class 对象。Class没有公用构造方法。Class对象是在加载类时由JVM以及经过调用类加载器中的defineClass方法自动构造的。继承
在深刻到反射机制以前,先探析一下反射机制的定义和应用。反射机制定义:Java反射机制是在运行状态时,对于任意一个类,都可以直到这个类的全部属性和方法;对于任意一个对象,都可以调用它的任意一个方法和属性。
在Java中,Class类和java.lang.reflect类库一块儿构成了对Java反射机制的支持。其中最常使用到的类是Constructor,Field,Method,而这三个类都继承了一个接口java.lang.reflect.Member。
实验类
//因篇幅缘由,实验代码未展现set,get等等经常使用方法 public class Cat { private String name = ""; private String master = ""; private int age = 0; public Cat() { } private Cat(String name, String master) { } public Cat(String name, String master, int age) { } public void eat() { System.out.println("小鱼干真好吃~"); } private void play() { System.out.println("快来陪我玩~"); } @Override public String toString() { return "Cat [name=" + name + ", master=" + master + ", age=" + age + "]"; } }
在java中获取Class对象有三种方法:
public static void main(String[] args) throws Exception { Cat c = new Cat(); Class cat1 = Class.forName("algorithms.sort.Cat"); Class cat2 = Cat.class; Class cat3 = c.getClass(); System.out.println(cat1 == cat2); System.out.println(cat2 == cat3); } //输出结果 true true
public static void main(String[] args) throws Exception { Class cat = Cat.class; //获取全部公共构造方法 Constructor<?> cons[] = cat.getConstructors(); for (Constructor<?> con : cons) { System.out.println("getConstructors-----------" + con); } System.out.println("**************************************"); //获取全部构造方法 Constructor<?> cons2[] = cat.getDeclaredConstructors(); for (Constructor<?> con2 : cons2) { System.out.println("getDeclaredConstructors---" + con2); } } //输出结果 getConstructors-----------public algorithms.sort.Cat(java.lang.String,java.lang.String,int) getConstructors-----------public algorithms.sort.Cat() ************************************** getDeclaredConstructors---public algorithms.sort.Cat(java.lang.String,java.lang.String,int) getDeclaredConstructors---private algorithms.sort.Cat(java.lang.String,java.lang.String) getDeclaredConstructors---public algorithms.sort.Cat()
建立对象
public static void main(String[] args) throws Exception { Class cat = Cat.class; //使用公共构造器实例化对象 Constructor<?> cons1 = cat.getConstructor(); //使用私有构造器实例化对象 Constructor<?> cons2 = cat.getDeclaredConstructor(String.class,String.class); Cat cat1 = (Cat)cons1.newInstance(); //私有的构造方法反射后要打开权限才能进行相应操做 cons2.setAccessible(true); Cat cat2 = (Cat)cons2.newInstance("tom","denny"); System.out.println(cat1); System.out.println(cat2); } //输出结果 Cat [name=, master=, age=0] Cat [name=tom, master=denny, age=0]
在建立对象的过程当中,值得注意的是若是反射的构造方法是私有的,那么要打开访问权限才能进行对象的实例化;也就是使用cons2.setAccessible(true);语句的缘由。
获取成员变量
public static void main(String[] args) throws Exception { Class cat = Cat.class; //获取构造器 Constructor<?> cons = cat.getConstructor(String.class,String.class,int.class); //实例化对象 Cat cat1 = (Cat)cons.newInstance("tom","denny",5); System.out.println(cat1); System.out.println("****************"); Field fields = cat.getDeclaredField("name"); //打开访问权限限制 fields.setAccessible(true); fields.set(cat1, "jack"); System.out.println(cat1); } //输出结果 Cat [name=tom, master=denny, age=5] **************** Cat [name=jack, master=denny, age=5]
获取成员方法
public static void main(String[] args) throws Exception { Class cat = Cat.class; //获取构造器 Constructor<?> cons = cat.getConstructor(String.class, String.class, int.class); //实例化对象 Cat cat1 = (Cat) cons.newInstance("tom", "denny", 5); System.out.println(cat1); System.out.println("****************"); //获取私有和公共成员方法 Method method1 = cat.getDeclaredMethod("setName", String.class); Method method2 = cat.getDeclaredMethod("eat"); method1.setAccessible(true); method1.invoke(cat1, "petter"); System.out.println(cat1); method2.invoke(cat1); } //输出结果 Cat [name=tom, master=denny, age=5] **************** Cat [name=petter, master=denny, age=5] 小鱼干真好吃~
public static void main(String[] args) throws Exception { //泛型只在编译期进行检查,在运行期会被擦除 ArrayList<Integer> list = new ArrayList<>(); list.add(111); list.add(222); //拿到字节码文件,字节码文件属于运行期 Class cla = Class.forName("java.util.ArrayList"); Method meth = cla.getMethod("add", Object.class); meth.invoke(list, "abc"); System.out.println(list); } //输出结果 [111, 222, abc]