欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

Java注解-自定义注解实例

程序员文章站 2022-05-28 19:10:28
...

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