JAVA基础之注解与反射的使用方法和场景

注解

注解定义

Java 注解(Annotation)又称 Java 标注,是 JDK5.0 引入的一种注释机制。
Java 语言中的类、方法、变量、参数和包等都可以被标注。和注释不同,Java 标注可以通过反射获取标注内容。在编译器生成类文件时,标注可以被嵌入到字节码中。Java 虚拟机可以保留标注内容,在运行 时可以获取到标注内容 。 当然它也支持自定义 Java 标注。

注解与注释的区别:注解是给机器看的注释,而注释是给程序员看的提示,编译时自动忽略注释。

使用场景

编译格式检查
反射中解析
生成帮助文档
跟踪代码依赖

内置注解

注解类型 注解意义 补充说明
@Override 重写 定义在java.lang.Override
@Deprecated 废弃 定义在java.lang.Deprecated
@SafeVarargs 忽略任何使用参数为泛型变量的方法或构造函数调用产生的警告 Java 7 开始支持
@FunctionalInterface 函数式接口 Java 8 开始支持,标识一个匿名函数或函数式接口。
@Repeatable 标识某注解可以在同一个声明上使用多次 Java 8 开始支持,标识某注解可以在同一个声明上使用多次。
@SuppressWarnings 抑制编译时的警告信息。 定义在java.lang.SuppressWarnings

*补充: @SuppressWarnings 的三种方法:

  • @SuppressWarnings(“unchecked”) [^ 抑制单类型的警告]
  • @SuppressWarnings(“unchecked”,“rawtypes”) [^ 抑制多类型的警告]
  • @SuppressWarnings(“all”) [^ 抑制所有类型的警告]

元注解

定义:

作用在其他注解的注解

元注解类型:

注解类型 注解意义
@Retention 标识这个注解怎么保存,是只在代码中,还是编入class文件中,或者是在运行时可以通过反射访问。
@Documented 标记这些注解是否包含在用户文档中 javadoc
@Target 标记这个注解应该是哪种 Java 成员
@Inherited 标记这个注解是自动继承的

*Inherited 补充:

  • 子类会继承父类使用的注解中被@Inherited修饰的注解
  • 接口继承关系中,子接口不会继承父接口中的任何注解,不管父接口中使用的注解有没有 被@Inherited修饰
  • 类实现接口时不会继承任何接口中定义的注解

自定义注解框架:

  • Annotation 与 RetentionPolicy 与 ElementType 。
    每 1 个 Annotation 对象,都会有唯一的 RetentionPolicy 属性;至于 ElementType 属性,则有 1~n个。
  • ElementType(注解的用途类型)“每 1 个 Annotation” 都与 “1~n 个 ElementType” 关联。当 Annotation 与某个 ElementType 关联 时,就意味着: Annotation有了某种用途。例如,若一个 Annotation 对象是 METHOD 类型,则该 Annotation 只能用来修饰方法。
package java.lang.annotation;
public enum ElementType {

TYPE,				/* 类、接口(包括注释类型)或枚举声明 */

FIELD,	 			/* 字段声明(包括枚举常量) */

METHOD,				/* 方法声明 */

PARAMETER,			/* 参数声明 */

CONSTRUCTOR,		/* 构造方法声明 */ 

LOCAL_VARIABLE,		/* 局部变量声明 */ 

ANNOTATION_TYPE, 	/* 注释类型声明 */

PACKAGE				 /* 包声明 */

}

RetentionPolicy(注解作用域策略)

“每 1 个 Annotation” 都与 “1 个 RetentionPolicy” 关联。

a) 若 Annotation 的类型为 SOURCE,则意味着:Annotation 仅存在于编译器处理期间,编译器 处理完之后,该 Annotation 就没用了。 例如," @Override" 标志就是一个 Annotation。当它修 饰一个方法的时候,就意味着该方法覆盖父类的方法;并且在编译期间会进行语法检查,编译器处理完后,"@Override" 就没有任何作用了。

b) 若 Annotation 的类型为 CLASS,则意味着:编译器将 Annotation 存储于类对应的 .class 文件 中,它是 Annotation 的默认行为。

c) 若 Annotation 的类型为 RUNTIME,则意味着:编译器将 Annotation 存储于 class 文件中,并 且可由JVM读入。

package java.lang.annotation;
public enum RetentionPolicy {
	SOURCE, /* Annotation信息仅存在于编译器处理期间,编译器处理完之后就没有该 Annotation信息了 */

	CLASS, /* 编译器将Annotation存储于类对应的.class文件中。默认行为 */

	RUNTIME /* 编译器将Annotation存储于class文件中,并且可由JVM读入 */
}

定义格式

@interface 自定义注解名{}

2.5.3、注意事项

  1. 定义的注解,自动继承了java.lang,annotation.Annotation接口
  2. 注解中的每一个方法,实际是声明的注解配置参数
    方法的名称就是 配置参数的名称
    方法的返回值类型,就是配置参数的类型。只能是:基本类型/Class/String/enum
  3. 可以通过default来声明参数的默认值
  4. 如果只有一个参数成员,一般参数名为value
  5. 注解元素必须要有值,我们定义注解元素时,经常使用空字符串、0作为默认值。

反射

JAVA反射机制是在运行状态中,获取任意一个类的结构 , 创建对象 , 得到方法,执行方法 , 属性;
这种在运行状态动态获取信息以及动态调用对象方法的功能被称为java语言的反射机制。

类加载器

Java类加载器(Java Classloader)是Java运行时环境(Java Runtime Environment)的一部分,
负责动态加载Java类到Java虚拟机的内存空间中。

java默认有三种类加载器,BootstrapClassLoader、ExtensionClassLoader、App ClassLoader。

  • BootstrapClassLoader(引导启动类加载器):嵌在JVM内核中的加载器,该加载器是用C++语言写的,主要负载加载JAVA_HOME/lib下的类库,引导启动类加载器无法被应用程序直接使用。ExtensionClassLoader(扩展类加载器):
  • ExtensionClassLoader是用JAVA编写,且它的父类加载器是Bootstrap。是由sun.misc.Launcher$ExtClassLoader实现的,主要加载JAVA_HOME/lib/ext目录中的类 库。它的父加载器是BootstrapClassLoaderApp ClassLoader(应用类加载器):
  • App ClassLoader是应用程序类加载器,负责加载应用程序classpath目录下的所有jar和class文 件。它的父加载器为Ext ClassLoader

类通常是按需加载,即第一次使用该类时才加载。由于有了类加载器,Java运行时系统不需要知道文件与 文件系统。学习类加载器时,掌握Java的委派概念很重要。

双亲委派模型:

如果一个类加载器收到了一个类加载请求,它不会自己去尝试加载这个类,而是把这个请求 转交给父类加载器去完成。每一个层次的类加载器都是如此。因此所有的类加载请求都应该传递到最顶层的 启动类加载器中,只有到父类加载器反馈自己无法完成这个加载请求(在它的搜索范围没有找到这个类) 时,子类加载器才会尝试自己去加载。委派的好处就是避免有些类被重复加载。

加载配置文件

给项目添加resource root目录

通过类加载器加载资源文件

默认加载的是src路径下的文件,但是当项目存在resource root目录时,就变为了加载 resource root下的文件了。

反射获取Class

要想了解一个类,必须先要获取到该类的字节码文件对象. 在Java中,每一个字节码文件,被夹在到内存后,都存在一个对应的Class类型的对象

得到Class

  • 如果在编写代码时, 知道类的名称, 且类已经存在, 可以通过
    包名.类名.class 得到一个类的 类对象如果拥有类的对象, 可以通过
  • Class 对象.getClass() 得到一个类的 类对象如果在编写代码时, 知道类的名称 , 可以通过
  • Class.forName(包名+类名): 得到一个类的 类对象

上述的三种方式, 在调用时, 如果类在内存中不存在, 则会加载到内存。如果类已经在内存中存在, 不会重复加载, 而是重复利用

特殊的类对象

基本数据类型的类对象:
基本数据类型.clss

包装类.type
基本数据类型包装类对象:
包装类.class

反射获取 Constructor

通过class对象 获取一个类的构造方法

1. 通过指定的参数类型, 获取指定的单个构造方法 getConstructor(参数类型的class对象数组)
例如:
构造方法如下:
Person(String name,int age)

得到这个构造方法的代码如下:
Constructor c = p.getClass().getConstructor(String.class,int.class);

2. 获取构造方法数组
getConstructors();

3. 获取所有权限的单个构造方法
getDeclaredConstructor(参数类型的class对象数组)

4. 获取所有权限的构造方法数组
getDeclaredConstructors();

Constructor 创建对象

newInstance(Object... para)
	调用这个构造方法, 把对应的对象创建出来
	参数: 是一个Object类型可变参数, 传递的参数顺序 必须匹配构造方法中形式参数列表的顺序

setAccessible(boolean flag)
	如果flag为true 则表示忽略访问权限检查 !(可以访问任何权限的方法)

反射获取 Method

通过class对象 获取一个类的方法

1. getMethod(String methodName , class.. clss)
	根据参数列表的类型和方法名, 得到一个方法(public修饰的)

2. getMethods();
	得到一个类的所有方法 (public修饰的)

3. getDeclaredMethod(String methodName , class.. clss)
	根据参数列表的类型和方法名, 得到一个方法(除继承以外所有的:包含私有, 共有, 保护, 默认)

4. getDeclaredMethods();
	得到一个类的所有方法 (除继承以外所有的:包含私有, 共有, 保护, 默认)

Method 执行方法

invoke(Object o,Object... para) :
	调用方法 ,
	参数1. 要调用方法的对象
	参数2. 要传递的参数列表

getName()
	获取方法的方法名称

setAccessible(boolean flag)
	如果flag为true 则表示忽略访问权限检查 (可以访问任何权限的方法)

反射获取 Field

通过class对象 获取一个类的属性

1. getDeclaredField(String filedName)
	根据属性的名称, 获取一个属性对象 (所有属性)
2. getDeclaredFields()
	获取所有属性
3. getField(String filedName)
	根据属性的名称, 获取一个属性对象 (public属性)
4. getFields()
	获取所有属性 (public)

Field 属性的对象类型

常用方法:
1.. get(Object o);
	参数: 要获取属性的对象 获取指定对象的此属性值

2. set(Object o , Object value);
	参数1. 要设置属性值的 对象
	参数2. 要设置的值 设置指定对象的属性的值

3. getName();
	获取属性的名称

4. setAccessible(boolean flag);
	如果flag为true 则表示忽略访问权限检查 !(可以访问任何权限的属性)

通过反射获取注解信息

获取类/属性/方法的全部注解对象

Annotation[] annotations01 = Class/Field/Method.getAnnotations();

for (Annotation annotation : annotations01) {
  System.out.println(annotation);
}

根据类型获取类/属性/方法的注解对象

注解类型 对象名 = (注解类型) c.getAnnotation(注解类型.class);

内省 Introspector

基于反射 , java所提供的一套应用到JavaBean的API

Bean类:
一个定义在包中的类 ,
拥有无参构造器
所有属性私有,
所有属性提供get/set方法
实现了序列化接口

Java提供了一套java.beans包的api , 对于反射的操作, 进行了封装

获取Bean类信息

方法:

BeanInfo getBeanInfo(Class cls)
通过传入的类信息, 得到这个Bean类的封装对象 .

获取bean类的 get/set方法 数组

常用的方法:
MethodDescriptor[] getPropertyDescriptors():

MethodDescriptor

常用方法:

1. Method getReadMethod();
	获取一个get方法
2. Method getWriteMethod();
	获取一个set方法

有可能返回null 注意 需要加判断

总结

到此这篇关于JAVA基础之注解与反射的使用方法和场景的文章就介绍到这了,更多相关JAVA注解与反射内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • java自定义注解验证手机格式的实现示例

    1.@Valid与@Validated的区别 1.1 基本区别 @Valid:Hibernate validation校验机制 @Validated:Spring Validator校验机制,这个也是最常用的 @Validation只是对@Valid进行了二次封装,在使用上并没有太大区别,但在分组.注解位置.嵌套验证等功能上有所不同 1.2 作用范围 @Validated:用在类型.方法和方法参数上.但不能用于成员属性(field) @Valid:可以用在方法.构造函数.方法参数和成员属性(fi

  • java 注解默认值操作

    我就废话不多说了,大家还是直接看代码吧~ package com.zejian.annotationdemo; import java.lang.annotation.ElementType; import java.lang.annotation.Retention; import java.lang.annotation.RetentionPolicy; import java.lang.annotation.Target; /** * Created by wuzejian on 2017

  • Java注解Annotation原理及自定义注解代码实例

    什么是注解? 对于很多初次接触的开发者来说应该都有这个疑问?Annontation是Java5开始引入的新特征,中文名称叫注解.它提供了一种安全的类似注释的机制,用来将任何的信息或元数据(metadata)与程序元素(类.方法.成员变量等)进行关联.为程序的元素(类.方法.成员变量)加上更直观更明了的说明,这些说明信息是与程序的业务逻辑无关,并且供指定的工具或框架使用. Annontation像一种修饰符一样,应用于包.类型.构造方法.方法.成员变量.参数及本地变量的声明语句中. Java注解是

  • Java中的注解和反射实例详解

    一.注解 注解(Annotation): 从jdk5.0开始引进,可以对程序进行解释或被其他程序读取. 注解格式:"@注释名",并可以添加一些参数. 例:@MyAnnotation(value='value') 1.内置注解 @override: 用于修饰方法,表示该方法声明是重写或实现一个父类的方法 @Deprecated: 用于修饰方法.属性.类,表示已过时不建议使用的 @SuppressWarnings: 用于抑制编译时的警告信息 2.元注解 作用:用于注解其他注解 @Targe

  • Java 自定义注解的魅力

    注解是什么? ①.引用自维基百科的内容: Java注解又称Java标注,是JDK5.0版本开始支持加入源代码的特殊语法 元数据 . Java语言中的类.方法.变量.参数和包等都可以被标注.和Javadoc不同,Java标注可以通过反射获取标注内容.在编译器生成类文件时,标注可以被嵌入到字节码中.Java虚拟机可以保留标注内容,在运行时可以获取到标注内容. 当然它也支持自定义Java标注. ②.引用自网络的内容: Java 注解是在 JDK5 时引入的新特性,注解(也被称为 元数据 )为我们在代码

  • 如何动态修改JavaBean中注解的参数值

    我这里有一个需求需要修改Person类中的一个属性上的注解的值进行修改,例如: public class Person { private int age; @ApiParam(access="lala") private String name; //get set 方法忽略 } 将@ApiParam(access="lala") 修改为@ApiParam(access="fafa"),经过分析是可以实现的,需要用到动态代理进行操作. 具体源码

  • 详解Java进阶知识注解

    一.注解的概念 1.注解官方解释 注解 叫元数据,一种代码级别的说明,它是JDK1.5及以后版本引入的一个特性,与类.接口.枚举在同一个层次,它可以声明在包.类.字段.局部变量.方法参数等的前面,用来对这些元素进行说明.注释. 注解的作用分类 编写文档:通过代码里表示的元数据生成文档[生成doc文档] 代码分析:通过代码里表示的元数据进行分析[使用反射] 编译检查:通过代码里表示的元数据让编译器能够实现基本的编译检查[Override] 注解按照运行机制分类 源码注解:注解只在源码中存在,编译成

  • Java元注解meta-annotation和依赖注入详解

    这篇文章既介绍一个技术,又记录一个逐渐探索发现的过程,以供大家参考. 缘起 注意到Java的依赖注入DI规范(起初以为是CDI规范,然后发现是DI规范)有个叫@Qualifier的注解,用于当一个interface或base class有多个实现类时,能选择其中一个实现.如不用这一注解,一般的(按类型)注入就会报错说"不知道要在多个实现中选哪一个".这一注解可以放在一个自定义注解上(例如@MyPreferredImplementation),从而将自定义注解变成一个qualifier

  • java注解的类型知识点总结

    提到java里的注解,和我们平时的注释还是有很大的区别,主要是作为java特性来使用的,跟我们常见的类是同一个使用的层面.关于java注解的类型,我们可以简单分为:自定义注解和元注解.其中元注解里的JDK又有5中注解的类型,下面一起来看看具体的内容讲解吧. 1.自定义注解 定义注解使用关键字: @interface // #1 定义注解 public @interface MyAnno1{ } 2.元注解 用于修饰注解的注解. JDK提供的5种元注解: (1)@Target:用于确定被修饰的自定

  • Java 注解学习笔记

    注解说明 Java注解又称Java标注,是Java语言5.0版本开始支持加入源代码的特殊语法元数据.为我们在代码中添加信息提供了一种形式化的方法,使我们可以在稍后某个时刻非常方便的使用这些数据.Java语言中的类.方法.变量.参数和包等都可以被标注.和Javadoc不同,Java标注可以通过反射获取注解内容.在编译器生成类文件时,注解可以被嵌入到字节码中.Java虚拟机可以保留注解内容,在运行时可以获取到注解内容. 内置注解 Java定义了一套注解,共有7个,3个在java.lang中,剩下4个

随机推荐