java注解前世今生

作者: hello_coke | 来源:发表于2016-09-10 17:49 被阅读570次

<h4>前序:</h4>
很开心下了班以后坐在电脑前面写这篇文章。注解Annotation我们常常见到jdk中的@override、@Deprecated、@SuppressWarnings 还有我本人认为非常优秀的开源项目spring中的IOC的@controller、@autowired和AOP的@Around @Before @After还有hibernate等我就不都枚举了。既然用的地方这么多所以我们就还是有必要揭开它的面纱看看它到底是怎么玩的。
<h4>1、属性全解</h4>
Java中提供了四种元注解,专门负责注解其他的注解,分别如下:
<pre>@Retention @Target @Documented @Inheried </pre>
先拿@override 的源码来作为一个切入点:
<pre>
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.SOURCE)
public @interface Override {
}
</pre>
@Target:说明了Annotation所修饰的对象范围,取值在一个枚举ElementType中,查看源码(局限于1.7。since1.8以后又新增了2个)
<pre>
public enum ElementType {
TYPE, 用于描述类、接口(包括注解类型) 或enum声明
FIELD, 成员变量、对象、属性(包括enum实例)
METHOD, 方法声明
PARAMETER, 参数声明
CONSTRUCTOR, 构造器声明
LOCAL_VARIABLE, 局部变量声明
ANNOTATION_TYPE, 元注解,有兴趣的可以看看@Retention、@Target、@Documented、@Inheried的@Target(ElementType.ANNOTATION_TYPE)都是如此
PACKAGE 包声明
}
</pre>
@Retention:表示需要在什么级别保存该注释信息(生命周期)。可选的RetentionPoicy参数包括:
<pre>
public enum RetentionPolicy {
SOURCE, 停留在java源文件,编译器被丢掉
CLASS, 停留在class文件中,但会被JVM丢弃(默认)
RUNTIME 内存中的字节码,JVM将在运行时也保留注解,因此可以通过反射机制读取注解的信息,所以我们自己开发注解的话一般都是RUNTIME,这样的话就可以通过注解来搞搞
}
</pre>
@Documented:一个简单的Annotations标记注解,表示是否将注解信息添加在java文档中。这个我们实际使用的会相对很少,因为我们大多会用在功能实现上。

@Inherited:指定Annotation具有继承性,上个例子:
<pre>
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Inherited
public @interface InheritedDemo {
}
//父类使用了Inherited注解
@InheritedDemo
public class Father {
}
//子类验证对于InheritedDemo注解是否有传递性
public class Child extends Father {
public static void main(String[] args) {
//输出为true
System.out.println(Child.class.isAnnotationPresent(InheritedDemo.class));
}
}
</pre>
<h4>2、编写自己的注解</h4>
记得最早期的时候在一家金融公司有用到注解的一个场景是,定义类中的每个属性的名称、类型及描述,好吧我们就开始着手写代码吧:
<pre>
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface FieldDesc {
//定义数据类型枚举
enum DataType{String,Integer,Long,Boolean,Double};
//名称
String name() ;
//数据类型
DataType dataType();
//中文描述
String desc() default "";
}
</pre>
<pre>
public class AnnationStudent {
@FieldDesc(name = "id", dataType = DataType.Integer, desc = "id")
private int id;
@FieldDesc(name = "name", dataType = DataType.String, desc = "名称")
private String name;
@FieldDesc(name = "age", dataType = DataType.Integer, desc = "年龄")
private int age;
@FieldDesc(name = "telNo", dataType = DataType.String, desc = "手机号码")
private String telNo;
public static void main(String[] args) {
Field[] fieldList = AnnationStudent.class.getDeclaredFields();
if (ArrayUtils.isNotEmpty(fieldList)) {
for (Field field : fieldList) {
Annotation[] annotationList = field.getAnnotations();
if (ArrayUtils.isNotEmpty(annotationList))
for (Annotation annotation : annotationList) {
if (annotation instanceof FieldDesc) {
FieldDesc desc = (FieldDesc) annotation;
System.out.println(desc.name() + " ### " + desc.dataType() + " ### " + desc.desc());
}
}
}
}
}
}


输出结果为:
id ### Integer ### id
name ### String ### 名称
age ### Integer ### 年龄
telNo ### String ### 手机号码
</pre>
这应该是一个完整的注解过程了吧。还有一个小技巧就是当我们给注解的属性赋值时,如果里面有个value属性,我们使用注解时可以不指定名称,自动会指定到value赋值:
<pre>
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
public @interface Test {
String value() default "";
String name() default "";
}
@Test("dadawd")
public class TE { <b>编译通过,如果Test 没有叫value的属性时编译会报错</b>
}
</pre>
这点在spring中的注解里面很常见。
<h4>3、谈谈注解在类似hibernate等框架中的应用</h4>
也许不一定是hibernate,但它们定义pojo实体的实现思想是一样的:
<pre>
@Entity
@Database(name="TestDB")
@Table(name="test_table")
public class TestEntity {
@Id
@Column(name="ID")
@GeneratedValue(strategy = GenerationType.AUTO)
@Type(value=Types.BIGINT)
private Long iD;</br>
@Column(name="Name")
@Type(value=Types.String)
private Long name;</br>
@Column(name="SourceType")
@Type(value=Types.INTEGER)
private Integer sourceType;</br>
public Long getiD() {
return iD;
}
public void setiD(Long iD) {
this.iD = iD;
}
public Long getName() {
return name;
}
public void setName(Long name) {
this.name = name;
}
public Integer getSourceType() {
return sourceType;
}
public void setSourceType(Integer sourceType) {
this.sourceType = sourceType;
}
}
</pre>
如何使用定义的@Database、@Table呢?
<pre>
//<b>获取数据库名</b>
public String getDatabaseName() {
Database db = clazz.getAnnotation(Database.class);
if (db != null && db.name() != null)
return db.name();
throw new RuntimeException("The entity must configure Database annotation.");
}
//<b>获取表名</b>
public String getTableName() {
Table table = clazz.getAnnotation(Table.class);
if (table != null && table.name() != null)
return table.name();
Entity entity = clazz.getAnnotation(Entity.class);
if ( entity != null && (!entity.name().isEmpty()) )
return entity.name();
return clazz.getSimpleName();
}
</pre>
这就是注解的使用所在。
<h4>4、结束语</h4>
我所理解的注解就是打标签下定义,比如我们常说某人是屌丝、高富帅、白富美,当知道某人是高富帅你又能反向推断出来是长的高、有钱、又帅这么一类人。

相关文章

  • java注解前世今生

    前序: 很开心下了班以后坐在电脑前面写这篇文章。注解Annotation我们常常见到jdk中的@override、...

  • 谈谈feign

    Feign的前世今生 Feign是什么? Feign的工作机制 处理注解 解析参数 构造http请求模板 发送请求...

  • Java的前世今生

    Java是什么?它是一门语言。其实和汉语,英语差不多,都是用于沟通的语言。 俗话说得好,逢人说人话,逢鬼说鬼话。比...

  • 讲讲ArrayList

    在之前《Java中的Collection》文章中简单粗略的介绍了Java中Collection前世今生及常用的Co...

  • 将军在上之男昭女惜重生三世千年孽缘

    前世!今生!来世再续! 前世欠谁!今生还!来世再续前缘! 前世因!今生续!来世果!

  • 前世今生来世缘

    谈何前世情 今生还 今生情 来世还 前世孽债 前世还 未了 今生还 今生欠 今生还 谈何来世还 来世欠 来世还 能...

  • 菜鸟学服务端技术----Spirng基础

    注解 Java基础加强总结(一)——注解(Annotation) java中的注解是如何工作的? java 注解 ...

  • JAVA日志的前世今生

    这世界上很多事情,看起来就像彩虹一样炫目而神奇,实际上背后蕴含着随处可见的原理。就好像静儿几年前买过一件超贵...

  • Java版本之前世今生

    一直比较仰慕Java两位大师(James Gosling 、Doug Lea),但是一直没有沉下心深入研究下Jav...

  • 自定义注解

    java annotation基础 java注解分为标准注解和元注解。 标准注解是java为我们提供的预定义的注解...

网友评论

    本文标题:java注解前世今生

    本文链接:https://www.haomeiwen.com/subject/vpqyettx.html