注解和反射
注解和反射
1、注解(Annotation)
作用:
不是程序本身,可以对程序作出解释
可以被其他程序(编译器)读取
格式:
注解是以“@注释名”在代码中存在的,还可以添加一些参数值
例如:@SuppressWarnings(value = "unlchecked").
在哪里使用:
可以附加在package、class、method、field等上面,相当于添加额外辅助信息
我们可以通过反射机制编程实现对这些元数据的访问
2、反射(Reflection)
反射是java被视为动态语言的关键,反射机制允许程序在执行期间借助于reflectionAPI取得任何类的内部信息,并能直接操作任意对象的内部属性及方法
Class c = Class.forName(“java.lang.String”)
加载完类之后,在对内存的方法区中就产生了一个Class类型的对象(一个类只有一个Class对象),这个对象就包含了完整的类的结构信息。我们可以通过这个对象看到类的结构,这个对象就是一个镜子,透过这个镜子看到类的结构,所以我们称之为反射
正常方式:引入需要的“包类”名称 ----> 通过new实例化 ------> 取得实例化对象
反射方式:实例化对象 ------> getClass()方法 ----->取得完整的包类名称
Java反射提供的功能:
- 在运行时判断任意一个对象所属的类
- 在运行时构造任意一个类的对象
- 在运行时判断任意一个类所具有的成员变量和方法
- 在运行时获取泛型信息
- 在运行时调用任意一个对象的成员变量和方法
- 在运行时处理注解
- 生成动态处理
一个类在内存只有一个class对象,一个类被加载后,类的整个结构都会被封装在class对象中
class类
对象照镜子后可以得到的信息:某个类的属性、左法和构造器、某个类到底实现了哪些接口。对于每个类而言,JRE都为其保留一个不变的Class类型的对象。一个Class对象包含了特定某个结构(class/interface/enum/annotation/primitive type/void/)的有关信息。
-
Class本身也是一个类
-
Class对象只能由系统建立对象
-
一个加载的类在JVM中只会有一个Class实例
-
一个Class对象对应的是一个加载到JVM中的一个.class文件
-
每个类的实例都会记得自己是由哪个Class实例所生成
-
通过Class可以完整地得到一个类中的所有被加载的结构
-
Class类是Reflection的根源,针对任何你想动态加载、运行的类,唯有先获得相应的class对象
获取Class类的实例
a)若已知具体的类,通过类的class属性获取,该方法最为安全可靠,程序性能最高。
Class clazz= Person.class;
b)已知某个类的实例,调用该实例的getClass()方法获取Class对象
Class clazz =person.getClass();
C)已知一个类的全类名,且该类在类路径下,可通过Class类的静态方法forName()获取,
可能抛出ClassNotFoundException
Class clazz= Class.forName("demo01.Student");
d)内置基本数据类型可以直接用类名.Type
什么时候会发生类初始化?
类的主动引用一定会发生类的初始化
当虚拟机启动,先初始化main方法所在的类>new一个类的对象
调用类的静态成员(除了final常量)和静态方法使用java.lang.reflect包的方法对类进行反射调用
当初始化一个类,如果其父类没有被初始化,则先会初始化它的父类类的被动引用(不会发生类的初始化)
当访问一个静态域时,只有真正声明这个域的类才会被初始化。如:当通过子类引用父类的静态变量,不会导
致子类初始化
通过数组定义类引用,不会触发此类的初始化
引用常量不会触发此类的初始化(常量在链接阶段就存入调用类的常量池中了)