最近又回顾了一下 Java 中的注解和反射知识点,注解在日常开发中使用很多,但是反射比较少。值得注意的是 Java 的各种框架底层源码中大量使用了注解和反射,阅读源码,这些是基本功,面试中这部分内容也经常问到。这里面概念不多,内容略微有些枯燥,但是通过一些简单的例子,能让我们明白一些基本概念和 API 的使用。所以,说到底,这篇博客只能算是一个简单的笔记,希望对你有帮助。 以前也写过枚举类和注解的相关笔记,可以看看历史文章 Java枚举类和注解梳理。
1、什么是注解
注解 Annotation 是从JDK1.5 开始引入的新技术。
注解的作用:不是程序本身,可以对程序作出解释,能被其他程序读取到。
注解使用的位置:package、class、method、field 等上面,相当于给他们添加了额外的辅助信息。我们可以通过反射机制实现对这些元数据的访问。
2、元注解
元注解的作用就是负责注解其他的注解,Java 定义了 4 个标准的 meta-Annotation类型,他们被用来提供对其他 Annotation 类型做说明。
Target:用于描述注解的使用范围,注解可以用在什么地方。随便点击一个注解,查看源码可以看到这个位置使用 ElementType枚举类表示,主要可以放在类上,方法上,属性上等,我这就不细说了。
Retention:表示在什么级别保存该注解的信息,用于描述注解的生命周期。SOURCE < CLASS
Document:说明该注解会被包含在 Javadoc中。
Inherited:说明子类可以继承父类的该注解。
3、自定义注解
使用 @interface 自定义注解时,自动继承了 java.lang.annotation.Annotation 接口。
分析:
@interface:用来声明一个注解,格式:public @interface 注解名{定义内容};
其中的每一个方法实际上就是一个配置参数;
方法的名称就是参数的名称;
返回的类型就是参数的类型(返回值只能是基本类型、Class、String、enum);
可以通过 default 来声明参数的默认值;
如果只有一个参数成员,一般参数名为 value;
注解元素必须要有值,我们定义注解元素时,经常使用空字符串,0作为默认值。
// 自定义注解
public class Test {
@MyAnnotation(age = 18, name = "Hello")
public void test() {}
}
@interface MyAnnotation {
// 注解的参数:参数类型 + 参数名();
String name() default "";
int age();
int id() default -1;
String[] schools() default {"清华大学", "北京大学"};
}
4、什么是反射
反射(Reflection):是 Java 被视为动态语言的关键,反射机制允许程序在执行期借助于 Reflection API 取得任何类的内部信息,并能直接操作任意对象的内部属性及方法。
加载完类后,在堆内存的方法区中就产生了一个 Class 类型的对象(一个类只有一个 Class 对象),这个对象包含了完整的类的结构信息。我们可以通过这个对象看到类的结构。这个对象就像一面镜子,透过和镜子看到类的结构。所以我们形象地称之为反射。
Java发射的优缺点:
优点:可以实现动态创建对象和编译,体现很大的灵活性;
缺点:对性能有影响,使用反射基本上是一种解释操作,这类操作总是慢于直接执行相同的操作。
public class Test01 {
public static void main(String[] args) throws ClassNotFoundException {
// 通过反射获取类的Class对象
Class c1 = Class.forName("Test01");
Class c2 = Class.forName("Test01");
System.out.println(c1); // class Test01
// 一个类在内存中只有一个Class对象
// 一个类被加载后,类的整个结构都会被封装在Class对象中。
System.out.println(c1.hashCode()); // 685325104
System.out.println(c2.hashCode()); // 685325104
}
}
class User{
private String name;
private int age;
public User() {}
public User(String name, int age) {
this.name = name;
this.age = age;
}
@Override
public String toString() {
return "User{" +
"name='" + name + '\'' +
", age=" + age +
'}';
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
}
5、Class 类及其创建方式
在Object类中定义了一下方法,此方法将被所有子类继承。
public final Class getClass()
此方法的返回值类型是一个Class类,此类是Java反射的源头,实际上所谓反射从程序的运行结果来看也很好理解:即&