在Java中注解随处可见,学习Java注解,知道其原理,可以读懂很多开源框架,如Spring,Mybatis等,还可以自定义注解实现更高级的功能。
一、常见的Java注解
Jdk自带的注解:@Override,@SuppressWarnings,@Deprecated(方法过时)
第三方框架注解:Spring,Mybatis等
二、注解的分类
1.按运行机制分
源码注解 源码存在,class文件不存在
编译时注解 源码,class文件存在
运行时注解 spring @antuAire
2.按来源分
Jdk自带的注解
第三方注解
自定义注解
3.元注解
给注解用的注解
三、注解语法
1.声明public @interface
2.成员以无参无异常方式声明
3.可以用default为成员指定一个默认值
int age() default 18;
4.成员的返回值类型是有限制的,合法的有基本数据类型,String,Class,Annotation,Enurmeration
5.如果注解只有一个成员,则成员名必须为value(),在使用时可以忽略成员名和赋值号(=)
6.注解类可以没有成员,没有成员的注解为标识注解
元注解
作用于注解上的注解,如@Target,@Retention,@Inherited,@Documented
package com.yuwl.ann; import java.lang.annotation.Documented; import java.lang.annotation.ElementType; import java.lang.annotation.Inherited; import java.lang.annotation.Retention; import java.lang.annotation.RetentionPolicy; import java.lang.annotation.Target; /** * 自定义注解 * @author Yuwl */ @Target({ElementType.METHOD,ElementType.TYPE}) @Retention(RetentionPolicy.RUNTIME) @Inherited @Documented public @interface Description { String value(); }
Target注解:注解的作用域,用在哪个地方,包含Java的所有元素:
CONSTRUCTOR 构造方法
FiELD 字段
LOCAL_VERIABLE 局部变量
METHOD 方法
PACKAGE 包
TYPE 类接口
Retention注解:生命周期,包含:
SOURCE 源码
CLASS 编译
RUNNTIME 运行时
Inheriter注解:标识性元注解,允许子类继承,但只适用于类的继承,不能用于接口继承,而且只会继承类的注解,不会继承方法的
Documented注解:生成javadoc时会包含注解
注解的使用:
@注解名(成员名1=成员值1,成员名2=成员值2)
四、自定义注解
1.自定义注解
package com.yuwl.ann; import java.lang.annotation.Documented; import java.lang.annotation.ElementType; import java.lang.annotation.Inherited; import java.lang.annotation.Retention; import java.lang.annotation.RetentionPolicy; import java.lang.annotation.Target; /** * 自定义注解 * @author Yuwl */ @Target({ElementType.METHOD,ElementType.TYPE}) @Retention(RetentionPolicy.RUNTIME) @Inherited @Documented public @interface Description { String value(); }
2.注解的使用
package com.yuwl.ann; /** * 自定义注解的使用 * @author Yuwl */ @Description("I am class annotation") public class UseAnnotation { @Description("I am method annotation") public void hello(){ } }
3.注解的解析
package com.yuwl.ann; import java.lang.annotation.Annotation; import java.lang.reflect.Method; /** * 解析注解 * @author Yuwl */ public class ParseAnnotation { public static void main(String[] args) { try { //1.使用类加载器加载类 Class c = Class.forName("com.yuwl.ann.UseAnnotation"); //2.找到类上的注解 boolean exist = c.isAnnotationPresent(Description.class); if(exist){ //3.拿到注解实例 Description d = (Description)c.getAnnotation(Description.class); System.out.println(d.value()); } //4.找到方法上的注解 Method[] ms = c.getMethods(); for(Method m : ms){ if(m.isAnnotationPresent(Description.class)){ Description d = (Description)m.getAnnotation(Description.class); System.out.println(d.value()); } } //5.方法注解的另一种解析方式 for(Method m : ms){ Annotation[] ans = m.getAnnotations(); for(Annotation an : ans){ Description d = (Description)an; System.out.println(d.value()); } } } catch (Exception e) { e.printStackTrace(); } } }
注解的解析主要用到Java的反射。
五、模拟实体到数据库表字段的映射
1.自定义表,字段注解
package com.yuwl.ann.dao; import java.lang.annotation.Documented; import java.lang.annotation.ElementType; import java.lang.annotation.Inherited; import java.lang.annotation.Retention; import java.lang.annotation.RetentionPolicy; import java.lang.annotation.Target; /** * 表自定义注解 * @author Yuwl */ @Target({ElementType.TYPE}) @Retention(RetentionPolicy.RUNTIME) @Inherited @Documented public @interface Table { String value(); }
package com.yuwl.ann.dao; import java.lang.annotation.Documented; import java.lang.annotation.ElementType; import java.lang.annotation.Inherited; import java.lang.annotation.Retention; import java.lang.annotation.RetentionPolicy; import java.lang.annotation.Target; /** * 字段自定义注解 * @author Yuwl */ @Target({ElementType.FIELD}) @Retention(RetentionPolicy.RUNTIME) @Inherited @Documented public @interface Column { String value(); }
2.实体使用注解
package com.yuwl.ann.dao; /** * 用户实体使用注解 * @author Yuwl */ @Table("user") public class User { @Column("id") private int id; @Column("userName") private String userName; @Column("sex") private int sex; @Column("mobile") private String mobile; public int getId() { return id; } public void setId(int id) { this.id = id; } public String getUserName() { return userName; } public void setUserName(String userName) { this.userName = userName; } public int getSex() { return sex; } public void setSex(int sex) { this.sex = sex; } public String getMobile() { return mobile; } public void setMobile(String mobile) { this.mobile = mobile; } }
3.测试
package com.yuwl.ann.dao; import java.lang.reflect.Field; import java.lang.reflect.Method; /** * 模拟实体到数据库表字段的测试 * @author Yuwl */ public class Test { public static void main(String[] args) { User u = new User(); u.setUserName("张三"); u.setSex(1); System.out.println(parseUser(u)); } public static String parseUser(User u){ StringBuffer sb = new StringBuffer(); sb.append("select * from "); try { //1.获取表名 Class c = Class.forName("com.yuwl.ann.dao.User"); if(c.isAnnotationPresent(Table.class)){ Table t = (Table)c.getAnnotation(Table.class); String tableName = t.value(); sb.append(tableName); } sb.append(" where 1=1"); //2.获取字段名与值 Field[] fs = c.getDeclaredFields(); for(Field f : fs){ //2.1字段名 String column = ""; if(f.isAnnotationPresent(Column.class)){ Column fld = (Column)f.getAnnotation(Column.class); column = fld.value(); } //2.2字段值 String fieldName = f.getName(); String getMethod = "get"+fieldName.substring(0, 1).toUpperCase()+fieldName.substring(1); Method method = c.getMethod(getMethod); Object fieldValue = method.invoke(u); if(fieldValue instanceof Integer && (Integer)fieldValue == 0){ continue; } if(fieldValue != null){ sb.append(" and ").append(column).append("="); if(fieldValue instanceof String){ sb.append("'").append(fieldValue).append("'"); }else{ sb.append(fieldValue); } } } } catch (Exception e) { e.printStackTrace(); } return sb.toString(); } }
效果:
select * from user where 1=1 and userName='张三' and sex=1
总结
Java注解不复杂,主要也就这么多东西,知道其实现原理,如何自定义注解,就能读懂别人写的注解,自己也能写了。