在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注解不复杂,主要也就这么多东西,知道其实现原理,如何自定义注解,就能读懂别人写的注解,自己也能写了。