注解与反射学习
一、注解
1.概念
Java 注解(Annotation)又称 Java 标注,是 JDK5.0 引入的一种注释机制。Java 语言中的类、方法、变量、参数和包等都可以被标注。和 Javadoc 不同,Java 标注可以通过反射获取标注内容。在编译器生成类文件时,标注可以被嵌入到字节码中。Java 虚拟机可以保留标注内容,在运行时可以获取到标注内容 。 当然它也支持自定义 Java 标注。
Java自带的注解是在java.lang.annotation包中,它有三个主干类,Annotation、RetentionPolicy、ElementType。
- **Annotation 就是个接口。**每 1 个 Annotation 对象,都会有唯一的 RetentionPolicy 属性;至于 ElementType 属性,则有 1~n 个。
- **ElementType 是 Enum 枚举类型,它用来指定 Annotation 的类型。**当 Annotation 与某个 ElementType 关联时,就意味着:Annotation有了某种用途。例如,若一个 Annotation 对象是 METHOD 类型,则该 Annotation 只能用来修饰方法。
- **RetentionPolicy 是 Enum 枚举类型,它用来指定 Annotation 的策略。**通俗点说,就是不同 RetentionPolicy 类型的 Annotation 的作用域不同。
2.类型
-
内置注解(作用在代码)
-
@Override - 检查该方法是否是重写方法。如果发现其父类,或者是引用的接口中并没有该方法时,会报编译错误。
-
@Deprecated - 标记过时方法。如果使用该方法,会报编译警告。
-
@SuppressWarnings - 指示编译器去忽略注解中声明的警告。带参数的,用法:@SuppressWarnings(value={“关键字”}),常用的关键字有
deprecation -- 使用了不赞成使用的类或方法时的警告 unchecked -- 执行了未检查的转换时的警告,例如当使用集合时没有用泛型 (Generics) 来指定集合保存的类型。 fallthrough -- 当 Switch 程序块直接通往下一种情况而没有 Break 时的警告。 path -- 在类路径、源文件路径等中有不存在的路径时的警告。 serial -- 当在可序列化的类上缺少 serialVersionUID 定义时的警告。 finally -- 任何 finally 子句不能正常完成时的警告。 all -- 关于以上所有情况的警告。
-
-
元注解(作用在其他注解)
-
@Retention - 标识这个注解怎么保存、在什么地方还有效,是只在代码中,还是编入class文件中,或者是在运行时可以通过反射访问。!默认是class策略(runtime>class>sources) 用法:@Retention(value =RetentionPolicy.RUNTIME)
意思就是指定该 Annotation 的策略是 RetentionPolicy.RUNTIME。这就意味着,编译器会将该 Annotation 信息保留在 .class 文件中,并且能被虚拟机读取。
-
@Documented - 标记这些注解是否包含在用户文档中。
-
@Target - 标记这个注解应该是哪种 Java 成员,即用在什么地方。
用法:@Target(ElementType.TYPE)
意思就是指定该 Annotation 的类型是 ElementType.TYPE。这就意味着,该注解是来修饰"类、接口(包括注释类型)或枚举声明"的注解。!默认情况指可以用于任何地方。
-
@Inherited - 标记这个注解是继承于哪个注解类(默认 注解并没有继承于任何子类)
-
-
自定义注解
用@interface来自定义注解,使用它都意味着实现了 java.lang.annotation.Annotation 接口,即该注解就是一个Annotation。定义 Annotation 时,@interface 是必须的。
**!注意:**它和我们通常的 implemented 实现接口的方法不同。Annotation 接口的实现细节都由编译器完成。通过 @interface 定义注解后,该注解不能继承其他的注解或接口。
另:自定义注解必须带有参数。 格式:参数类型+参数名(); (**注:**这个括号必须有)
例:
public class Test03 { @MyAnnotation3(name = "小宇",age = 18,city = {"广州","深圳","上海"}) public void test(){ } } @Target({ElementType.TYPE,ElementType.METHOD}) @Retention(RetentionPolicy.CLASS) @interface MyAnnotation3{ //注解参数:参数类型 + 参数名(); String name() default "";//default 表示参数默认值,这样使用该注解时,不不设置参数也不报错 int age(); String[] city(); }
二、反射
1.概述
(1)Java反射机制的核心是在程序运行时动态加载类并获取类的详细信息,从而操作类或对象的属性和方法。即允许程序在执行期间借助于Reflection API取得任何类的内部消息,并直接操作任意对象的内部属性及方法。正是反射机制使得Java由静态语言变为准动态语言!
**拓展:**主要动态语言有Object-C、C#、JavaScript、PHP、Python等:主要静态语言有:Java、C、C++。
(2)加载完类之后,在堆内存的方法区中就产生了一个Class类型的对象(一个类只有一个Class对象),这个对象就包含了完整的类的结构信息。通过反射,可以在运行时动态地创建对象并调用其属性。
通过代码来认识一下!
//什么叫反射
public class Test01 {
public static void main(String[] args) throws ClassNotFoundException {
//通过反射来获取类的Class对象
Class<?> c1 = Class.forName("com.yg.reflection.User");//相对路径下类的位置
System.out.println(c1);
Class<?> c2 = Class.forName("com.yg.reflection.User");
Class<?> c3 = Class.forName("com.yg.reflection.User");
Class<?> c4 = Class.forName("com.yg.reflection.User");
//打印hash值,验证相等
//得出结论:一个类在内存中只有一个Class对象;一个类别加载后,类的整个结构都被封装在Class对象中
System.out.println(c2.hashCode());
System.out.println(c3.hashCode());
System.out.println(c4.hashCode());
}
}
//实体类:称为pojo或entity
class User{
}
结果图:
2.优缺点
-
优:
- 在运行时获得类的各种内容,进行反编译
- 方便创建灵活的代码,避免将程序写死到代码里。
- 无需在组件之间进行源代码的链接,更容易实现面向对象
-
缺:
- 性能问题。主要应用在对灵活性和扩展性要求很高的系统框架上,普通程序不建议使用。
- 安全问题。反射调用方法时可以忽略权限检查,因此可能会破坏封装性
- 维护问题。程序人员希望在源代码中看到程序的逻辑,反射绕过了源代码的技术,代码比相应的直接代码更复杂。
3.类的加载与内存分析
1)类加器内存分析
- **加载:**将class文件字节码内容加载到内存中,并将这些静态数据转换成方法区的运行时数据结构,然后生成一个代表这个类的java.lang.Class对象.
- **链接:**将Java类的二进制代码合并到jvm的运行状态之中的过程。
- *验证:*确保加载的类信息符合jvm规范,没有安全方面的问题
- *准备:*正式为类变量(static)分配内存并设置类变量默认初始值的阶段,这些内存都将在方法区中进行分配。
- *解析:*虚拟机常量池内的符号引用(常量名)替换为直接引用(地址)的过程。
- **初始化:**为类的静态变量赋予正确的初始值
- 执行类构造器()方法的过程。类构造器()方法是由编译期自动收集类中所有类变量的赋值动作和静态代码块中的语句合并产生的。(类构造器是构造类信息的,不是构造该类对象的构造器)。
- 当初始化一个类的时候,如果发现其父类还没有进行初始化,则需要先触发其父类的初始化。
- 虚拟机会保证一个类的()方法在多线程环境中被正确加锁和同步。
示例:
//测试类加载内存分析
public class Test04 {
public static void main(String[] args) {
A a = new A();
System.out.println(A.m);
/*
1.加载到内存,会产生一个类对应Class对象
2.链接,链接结束后m = 0;
3.初始化
<clinit>(){
System.out.println("A类静态代码块初始化");
m = 300;
m = 100;
}
最终
m =100
*/
}
}
//定义一个类
class A{
static {
System.out.println("A类静态代码块初始化");
m = 300;
}
static int m = 100;
//静态代码块先执行,首先是m=300,然后才是m=100,即最终是m=100.
public A() {
System.out.println("A类的无参构造代码块");
}
}
结果图:
- **初始化补充!**类的初始化什么时候会发生呢?
- 类的主动引用(一定会发生类的初始化)
- 当虚拟机启动,先初始化main方法所在的类(注意!)
- new一个类的对象
- 调用类的静态成员(除了final常量)和静态方法
- 使用java.lang.reflect包的方法对类进行反射调用
- 当初始化一个类,如果其父类没有被初始化,则先会初始化它的父类
- 类的被动引用(不会发生类的初始化)
- 当访问一个静态域时,只有真正声明这个域的类才会被初始化。如:当通过子类引用父类的静态变量,不会导致子类初始化
- 通过数组定义类引用,不会触发此类的初始化
- 引用常量不会触发此类的初始化(常量在链接阶段就存入调用类的常量池中了)(注意!)
- 类的主动引用(一定会发生类的初始化)
示例:
//测试什么时候类会被初始化
public class Test05 {
static {
System.out.println("Main类被加载");
}
public static void main(String[] args) throws ClassNotFoundException {
//1.主动引用(会发生类的初始化)
//父类没被初始化
//Son son = new Son();
//反射也会产生主动引用
//Class.forName("com.yg.reflection.Son");
//2.被动引用(不会发生类的初始化)
//访问静态域
//System.out.println(Son.b);
//通过数组引用
//Son[] array = new Son[10];
//引用常量(常量在链接阶段就存入调用类的常量池中了)
System.out.println(Son.M);
}
}
//定义父类
class Father{
static {
System.out.println("父类被加载");
}
static int b = 2;
}
//定义子类
class Son extends Father{
static {
System.out.println("子类被加载");
m = 300;
}
static int m = 100;
static final int M = 2;
2)类加载器的作用
- **作用:**将class文件字节码内容加载到内存中,并将这些静态数据转换成方法区的运行时数据结构,然后在堆中生产一个代表这个类的Java.lang.Class对象,作为方法区中类数据的访问入口。
- **分类:**分为两类,一类是系统提供的,另外一类则是由 Java 应用开发人员编写的。
- **引导类加载器(bootstrap class loader)也称根类加载器:**它用来加载 Java 的核心库(jre/lib/rt.jar),是用原生C++代码来实现的,并不继承自java.lang.ClassLoader。加载扩展类和应用程序类加载器,并指定他们的父类加载器,在java中获取不到。
- **扩展类加载器(extensions class loader):**它用来加载 Java 的扩展库(jre/ext/*.jar)。Java 虚拟机的实现会提供一个扩展库目录。该类加载器在此目录里面查找并加载 Java
- **系统类加载器(system class loader):**它根据 Java 应用的类路径(classPath)来加载 Java 类。一般来说,Java 应用的类都是由它来完成加载的。可以通过 ClassLoader.getSystemClassLoader()来获取它。
- **自定义类加载器(custom class loader):**除了系统提供的类加载器以外,开发人员可以通过继承 java.lang.ClassLoader类的方式实现自己的类加载器,以满足一些特殊的需求。(为第二类!)
示例:
//类加载器的作用
public class Test06 {
public static void main(String[] args) throws ClassNotFoundException {
//获取类的加载器
ClassLoader systemClassLoader = ClassLoader.getSystemClassLoader();
System.out.println(systemClassLoader);
//获取系统类加载器的父类加载器--》扩展类加载器
ClassLoader parent = systemClassLoader.getParent();
System.out.println(parent);
//获取扩展类加载器的父类加载器--》根加载器
ClassLoader parent1 = parent.getParent();
System.out.println(parent1);//是用原生C/C++代码来实现的,并不继承自java.lang.ClassLoader,在java中获取不到。
//测试当前(即自定义类)类是哪个加载器加载
ClassLoader classLoader = Class.forName("com.yg.reflection.Test06").getClassLoader();
System.out.println(classLoader);
//测试JDK内置的类是谁加载的
ClassLoader classLoader1 = Class.forName("java.lang.Object").getClassLoader();
System.out.println(classLoader1);
//如何获得系统类加载器可以加载的路径
System.out.println(System.getProperty("java.class.path"));
}
}
结果图:
可以看出内置的类是由根类加载器加载的,Java无法获取!
4.反射的使用
1)获取Class的三方法
-
方法一:对象·getClass()
-
方法二:任何类型·class
-
方法三:Class.forName(“完整类名带包名”) (通过class类的静态方法,最常用!)
-
方法四(补充):Integer.TYPE,基本内置类型的包装类都有一个Type属性
示例:
//获取class类的方法 public class Test02 { public static void main(String[] args) throws ClassNotFoundException { Person person = new Student(); System.out.println("这个人是:"+person.name); //方法一:通过对象获得 Class c1 = person.getClass(); System.out.println(c1.hashCode()); //方法二:通过类名获得 Class c2 = Student.class; System.out.println(c2.hashCode()); //方法三:forName获得 Class c3 = Class.forName("com.yg.reflection.Student"); System.out.println(c3.hashCode()); //方法四:基本内置类型的包装类都有一个Type属性 Class c4 = Integer.TYPE; System.out.println(c4.hashCode()); //获得父类类型 Class c5 = c1.getSuperclass(); System.out.println(c5); } } //定义一个父类 class Person{ String name; public Person() { } public Person(String name) { this.name = name; } @Override public String toString() { return "Person{" + "name='" + name + '\'' + '}'; } } //定义子类 class Student extends Person{ public Student() { this.name = "学生"; } }
2)Class对象的类型
-
Object.class–>类
-
Comparable.class–>接口
-
String[].class–>一维数组
-
int[] [].class–>二维数组
-
Override.class–>注解
-
ElementType.class–>枚举
-
Integer.class–>基本数据类型
-
void.class–>void
-
Class.class–>Class类型
示例:
//所有类型的Class对象 public class Test03 { public static void main(String[] args) { Class c1 = Object.class;//类 Class c2 = Comparable.class;//接口 Class c3 = String[].class;//一维数组 Class c4 = int[][].class;//二维数组 Class c5 = Override.class;//注解 Class c6 = ElementType.class;//枚举 Class c7 = Integer.class;//基本数据类型 Class c8 = void.class;//void Class c9 = Class.class;//Class类型 System.out.println(c1); System.out.println(c2); System.out.println(c3); System.out.println(c4); System.out.println(c5); System.out.println(c6); System.out.println(c7); System.out.println(c8); System.out.println(c9); //只要元素类型与维度一样,就是同一个Class int[] a = new int[10]; int[] b = new int[100]; System.out.println(a.getClass().hashCode()); System.out.println(b.getClass().hashCode()); } }
结果图:
3)获取类运行时的信息
步骤:
-
获取Class对象
-
通过对象调用get或getDeclared类方法来获取信息
-
getName()–>获取包名+类名
-
getFields()–>获取public属性
-
getMethods()–>获得本类极其父类的全部public方法
-
getConstructors()–>获得public权限的构造器
上面都有相对应的getDeclared类方法,该类方法都是可以获得需求有的全部!
-
示例:
//获取类的信息
public class Test07 {
public static void main(String[] args) throws ClassNotFoundException, NoSuchFieldException, NoSuchMethodException {
Class c1 = Class.forName("com.yg.reflection.User");
//1.获取类的名字
System.out.println(c1.getName());//获取包名+类名
System.out.println(c1.getSimpleName());//获取类名
System.out.println("==========================================");
//2.获取类的属性
Field[] fields = c1.getFields();//只能找到public属性
for (Field field : fields) {
System.out.println(field);
}
Field[] declaredFields = c1.getDeclaredFields();//找到全部属性
for (Field declaredField : declaredFields) {
System.out.println(declaredField);
}
//获得指定属性的值
Field name = c1.getDeclaredField("name");
System.out.println(name);
System.out.println("==========================================");
//3.获取类的方法
Method[] methods = c1.getMethods();//获得本类极其父类的全部public方法
for (Method method : methods) {
System.out.println("正常的:"+method);
}
Method[] declaredMethods = c1.getDeclaredMethods();//获得本类的所有方法
for (Method declaredMethod : declaredMethods) {
System.out.println("getDeclaredMethods: "+declaredMethod);
}
//获得指定的方法
Method getName = c1.getDeclaredMethod("getName", null);
Method setName = c1.getDeclaredMethod("setName", String.class);
System.out.println(getName);
System.out.println(setName);
System.out.println("==========================================");
//4.获取类的构造器
Constructor[] constructors = c1.getConstructors();//只能获得public
for (Constructor constructor : constructors) {
System.out.println(constructor);
}
Constructor[] declaredConstructors = c1.getDeclaredConstructors();//获得本类全部
for (Constructor declaredConstructor : declaredConstructors) {
System.out.println(declaredConstructor);
}
//获得指定的构造器
Constructor declaredConstructor = c1.getDeclaredConstructor(String.class, int.class, int.class);
System.out.println("指定:"+declaredConstructor);
}
}
4)动态创建对象
-
创建类的对象:调用Class对象的newInstance()方法
- 1.类必须有一个无参数的构造器
- 2.类的构造器访问权限需要足够
-
若类没有无参构造器,则使用下面方法
- 1.通过getDeclaredConstructor()方法取得本来的指定形参类型的构造器
- 2.向构造器中传递一个对象组进去,里面包含了构造器中所需的各个参数
- 3.通过Constructor实例化对象
-
调用指定的方法,通过反射
-
1.通过getMethod()方法取得一个Method对象,并设置此方法操作时所需要的参数类型
-
2.使用Object.invoke(Object obj, Object[] args)进行调用,并向方法中传递要设置的obj对象的参数信息
!注意:若原方法声明为private,则需要在调用此invoke()方法前,调用方法对象的setAccessible(true)方法,将可访问private的方法。
-
-
调用操作指定的属性的话,步骤同调用指定的方法,小小的改变就是Object.invoke()方法改为Object.set()方法。
5)反射操作泛型
- Java采用泛型擦除机制来引入泛型。Java中的泛型仅仅是给编译器Javac使用的,确保数据的安全性和免去强制类型转换的麻烦。但是编译一旦完成,所有和泛型有关的类型全部被擦除。
- 为了通过反射操作这些类型以迎合实际开发的需要,Java新增ParameterizedType ,GenericArrayType,TypeVariable和WildcardType几种类型来代表不能被归一到Class类中的类型但是又和原始类型齐名的类型。
- ParameterizedType:表示一种参数化的类型,比如Collection< String >
- GenericArrayType:表示一种元素类型是参数化类型或者类型变量的数组类型
- TypeVariable:是各种类型变量的公共父接口
- WildcardType:代表一种通配符类型表达式
示例:
//通过反射获取泛型
public class Test10 {
//定义两个测试方法
public void test01(Map<String,User>map, List<User>list){
System.out.println("test01");
}
public Map<String,User> test02(){
System.out.println("test02");
return null;
}
public static void main(String[] args) throws NoSuchMethodException {
//通过反射获取泛型
Method method = Test10.class.getMethod("test01", Map.class, List.class);
Type[] genericParameterTypes = method.getGenericParameterTypes();
for (Type genericParameterType : genericParameterTypes) {
System.out.println("#"+genericParameterType);
if (genericParameterType instanceof ParameterizedType){
Type[] actualTypeArguments = ((ParameterizedType) genericParameterType).getActualTypeArguments();
for (Type actualTypeArgument : actualTypeArguments) {
System.out.println(actualTypeArgument);
}
}
}
method = Test10.class.getMethod("test02",null);
Type genericReturnType = method.getGenericReturnType();
if (genericReturnType instanceof ParameterizedType){
Type[] actualTypeArguments = ((ParameterizedType)genericReturnType).getActualTypeArguments();
for (Type actualTypeArgument : actualTypeArguments) {
System.out.println(actualTypeArgument);
}
}
}
}
6)反射操作注解
-
获取注解的方法
- Class.getAnnotations() 获取所有的注解,包括自己声明的以及继承的
- Class.getAnnotation(Class< A > annotationClass) 获取指定的注解,该注解可以是自己声明的,也可以是继承的
- Class.getDeclaredAnnotations() 获取自己声明的注解
-
了解ORM
什么是ORM?
对象关系映射(Object Relational Mapping,简称ORM)模式是一种为了解决面向对象与关系数据库存在的互不匹配的现象的技术。简单的说,ORM是通过使用描述对象和数据库之间映射的元数据,将程序中的对象自动持久化到关系数据库中。具体更深理解嘚到数据库的知识点!
-
通过注解获得反射简单模拟ORM
例:
//练习反射操作注解 public class Test11 { public static void main(String[] args) throws ClassNotFoundException, NoSuchFieldException { Class c1 = Class.forName("com.yg.reflection.Student2"); //通过反射获得注解 Annotation[] annotations = c1.getAnnotations(); for (Annotation annotation : annotations) { System.out.println(annotation); } //通过反射获得注解value的值 TableYu tableYu = (TableYu)c1.getAnnotation(TableYu.class); String value = tableYu.value(); System.out.println(value); System.out.println("----------------------------"); //通过反射获得类指定的注解 Field f_name = c1.getDeclaredField("id"); FieldYu annotation = f_name.getAnnotation(FieldYu.class); System.out.println(annotation.columnName()); System.out.println(annotation.type()); System.out.println(annotation.length()); } } @TableYu("db_student") class Student2{ @FieldYu(columnName = "db_id",type ="int",length = 10) private int id ; @FieldYu(columnName = "db_age",type ="int",length = 10) private int age; @FieldYu(columnName = "db_name",type ="varchar",length = 3)//数据库里面,字符用varchar代替String private String name; public Student2(int id, int age, String name) { this.id = id; this.age = age; this.name = name; } public int getId() { return id; } public void setId(int id) { this.id = id; } public int getAge() { return age; } public void setAge(int age) { this.age = age; } public String getName() { return name; } public void setName(String name) { this.name = name; } @Override public String toString() { return "Student2{" + "id=" + id + ", age=" + age + ", name='" + name + '\'' + '}'; } } //类名的注解 @Target(ElementType.TYPE) @Retention(RetentionPolicy.RUNTIME) @interface TableYu{ String value(); } //属性的注解 @Target(ElementType.FIELD) @Retention(RetentionPolicy.RUNTIME) @interface FieldYu{ String columnName(); String type(); int length(); }
结果图:
注解和反射知识,建议一直学习,会学得更透!本次学习笔记分享到此!