JAVA自定义注解(1)

1、自定义注解@People

package com.kute.test.selfannotation;

import java.awt.Color;
import java.lang.annotation.Documented;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

/**
 * 使用@interface自定义注解时,自动继承了java.lang.annotation.Annotation接口,由编译程序自动完成其他细节。
 * 在定义注解时,不能继承其他的注解或接口。
 * @interface用来声明一个注解,其中的每一个方法实际上是声明了一个配置参数。
 * 方法的名称就是参数的名称,返回值类型就是参数的类型(返回值类型只能是基本类型、Class、String、enum)。
 * 可以通过default来声明参数的默认值,如果不指定default,则该参数为必设项
 * 
 * Annotation类型里面的参数该怎么设定: 
 * 第一,只能用public或默认(default)这两个访问权修饰.例如,String value();这里把方法设为defaul默认类型;
 * 第二,参数成员只能用基本类型byte,short,char,int,long,float,double,boolean八种基本数据类型
 * 	       和 String,Enum,Class,annotations等数据类型以及这一些类型的数组.
 * 第三,如果只有一个参数成员,最好把参数名称设为"value",后加小括号.
 * 
 * 注解元素必须有确定的值,要么在定义注解的默认值中指定,要么在使用注解时指定,非基本类型的注解元素的值不可为null。
 * 因此, 使用空字符串或0作为默认值是一种常用的做法。这个约束使得处理器很难表现一个元素的存在或缺失的状态,
 * 因为每个注解的声明中,所有元素都存在,并且都具有相应的值,为了绕开这个约束,我们只能定义一些特殊的值["",-1]等,
 * 例如空字符串或者负数,一次表示某个元素不存在,在定义注解时,这已经成为一个习惯用法。
 * 
 * 定义了注解,并在需要的时候给相关类,类属性加上注解信息,如果没有响应的注解信息处理流程,
 * 注解可以说是没有实用价值。如何让注解真真的发挥作用,主要就在于注解处理方法
 * 
 * @Target注解参数
 * 1.CONSTRUCTOR:构造函数
 * 2.FIELD:字段,枚举常量
 * 3.LOCAL_VARIABLE:局部变量
 * 4.METHOD:方法
 * 5.PACKAGE:包
 * 6.PARAMETER:方法参数
 * 7.TYPE:接口,类,枚举,注解
 * 
 * 下面就开始定义一个@People的注解
 */

@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.TYPE,ElementType.METHOD})
public @interface People {
	
	//定义一个枚举,(男人,女人,好人,坏人,大人,小人)
	public enum PeopleType{MAN, WOMAN, GOODMAN, BADMAN, BIGMAN, SMALLMAN}
	//定义了一个注解参数:人的类型,默认为男人
	public PeopleType type() default PeopleType.MAN;
	
	public String name() default "";
	//此参数没有指定default,所以用@People注解的时候必须指定该参数
	public int age();
	
	public String[] fruitColor() default { "red", "black" };
	
}

2、解析注解

package com.kute.test.selfannotation;

import java.lang.reflect.Method;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.BeansException;
import org.springframework.beans.factory.InitializingBean;
import org.springframework.beans.factory.config.BeanPostProcessor;
import org.springframework.beans.factory.config.PropertyPlaceholderConfigurer;
import org.springframework.util.ReflectionUtils;

import com.kute.test.selfannotation.People.PeopleType;

public class AnnotionParse {

	private static Logger logger = LoggerFactory.getLogger(AnnotionParse.class);

	public void parse(Class clazz) {
		try {
			Object clasz = clazz.getConstructor(new Class[] {}).newInstance(
					new Object[] {});
			// clasz类上是否标注有People这个注解
			if (null != clasz.getClass().getAnnotation(People.class)) {
				Method[] methods = clasz.getClass().getDeclaredMethods();
				for (Method method : methods) {
					// annotation是方法上的注解
					People people = method.getAnnotation(People.class);
					if (null != people) {
						// 得到注解上的type参数的值
						Object oType = people.type();
						Object oName = people.name();
						int oAge = people.age();
						String[] fruitColor = people.fruitColor();

						StringBuffer buffer = new StringBuffer();
						for (String s : fruitColor)
							buffer.append(s + ",");

						logger.info("取出了方法的参数:[oType:" + oType + "],[oName:"
								+ oName + "],[oAge:" + oAge + "],[fruitColor:"
								+ buffer.toString() + "]");

						// 判断oType的类型是否是要指定的类型
						if (method.getParameterTypes()[0]
								.equals("com.kute.test.selfannotation.People.PeopleType")) {
							oType = (PeopleType) oType;
						}
						if (method.getParameterTypes()[1]
								.equals("java.lang.String")) {
							oName = oName.toString();
						}
						// 反射调用方法并传递参数
						ReflectionUtils.invokeMethod(method, clasz,
								new Object[] { oType, oName, oAge, fruitColor });
					}
				}
			}
		} catch (Exception e) {
			e.printStackTrace();
		}
	}
}

3、应用注解

package com.kute.test.selfannotation;

import java.io.Serializable;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import com.kute.test.selfannotation.People.PeopleType;

@People(age = 20)
public class Student implements Serializable {

	private static final long serialVersionUID = 1L;
	private static Logger logger = LoggerFactory.getLogger(Student.class);

	@People(type = PeopleType.GOODMAN, name = "kute", age = 18, fruitColor = {
			"blue", "white", "green" })
	public void setValue(PeopleType stuType, String stuName, int stuAge,
			String[] fruitColor) {

		StringBuffer buffer = new StringBuffer();
		for (String color : fruitColor)
			buffer.append(color + ",");
		logger.info("成功调用了方法并注入了参数:[" + stuType + "," + stuAge + "," + stuName
				+ ", {" + buffer.toString() + "} ]");
	}

}

4、测试

package test;

import org.junit.Test;

import com.kute.test.selfannotation.AnnotionParse;
import com.kute.test.selfannotation.Student;

public class TestAnnotationParse {
	
	@Test
	public void test() {
		AnnotionParse parse = new AnnotionParse();
		parse.parse(Student.class);
	}
}

5、运行结果

28 [main] INFO com.kute.test.selfannotation.AnnotionParse - 取出了方法的参数:[oType:GOODMAN],[oName:kute],[oAge:18],[fruitColor:blue,white,green,]
28 [main] INFO com.kute.test.selfannotation.Student - 成功调用了方法并注入了参数:[GOODMAN,18,kute, {blue,white,green,} ]


此例熟悉之后,下面还有个实际中应用比较广泛的自定义注解的例子:http://blog.csdn.net/kutekute/article/details/17260771


  • 2
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值