元注解:
1.@Target:作用:用于描述注解的使用范围(即:被描述的注解可以用在什么地方)
取值(ElementType)有:
1.CONSTRUCTOR:用于描述构造器
2.FIELD:用于描述域
3.LOCAL_VARIABLE:用于描述局部变量
4.METHOD:用于描述方法
5.PACKAGE:用于描述包
6.PARAMETER:用于描述参数
7.TYPE:用于描述类、接口(包括注解类型) 或enum声明
2.@Retention:作用:表示需要在什么级别保存该注释信息,用于描述注解的生命周期(即:被描述的注解在什么范围内有效)
1.SOURCE:在源文件中有效(即源文件保留)
2.CLASS:在class文件中有效(即class保留) 即 编译时注解
3.RUNTIME:在运行时有效(即运行时保留) 即 运行时注解
3.@Documented: @Documented用于描述其它类型的annotation应该被作为被标注的程序成员的公共API,因此可以被例如javadoc此类的工具文档化。Documented是一个标记注解,没有成员。
4.@Inherited:@Inherited 元注解是一个标记注解,@Inherited阐述了某个被标注的类型是被继承的。如果一个使用了@Inherited修饰的annotation类型被用于一个class,则这个annotation将被用于该class的子类。
常用注解一般是 运行时注解 和编译时注解,二运行时注解是很影响效率的,所有常用框架 比如ButterKnife 都是编译时注解,在程序编译的时候生成代码完成注解。
自定义注解
注解处理器(Annotation Processor) ,是javac的一个工具, 用来在编译时扫描和处理注解,所以在自定义注解的时候,一般需要自定义一个 注解处理器来 处理相关注解逻辑
@UserAnnotation(age=20,gender="F",id=2014,name="zhangsan")//注解的使用
private Object obj;
public static void main(String[] args) throws Exception
{
Filed objField = TestMain.class.getField("obj");
UserAnnotation ua = objField.getAnnotation(UserAnnotation.class);//得到注解,起到了标记的作用
System.out.println(ua.age()+","+ua.gender()+","+ua.id()+","+ua.name());
实现注解处理器,要继承 AbstractProcessor
@AutoService(Processor.class)
public class CustomProcessor extends AbstractProcessor {
/**
* 注解处理器的初始化
* 一般在这里获取我们需要的工具类
* @param processingEnvironment 提供工具类Elements, Types和Filer
*/
@Override
public synchronized void init(ProcessingEnvironment env){
super.init(env);
//Element代表程序的元素,例如包、类、方法。
mElementUtils = env.getElementUtils();
//处理TypeMirror的工具类,用于取类信息
mTypeUtils = env.getTypeUtils();
//Filer可以创建文件
mFiler = env.getFiler();
//错误处理工具
mMessages = env.getMessager();
}
/**
* 处理器实际处理逻辑入口
* @param set
* @param roundEnvironment 所有注解的集合
* @return
*/
@Override
public boolean process(Set<? extends TypeElement> annoations,
RoundEnvironment env) {
//do someThing
}
//指定注解处理器是注册给哪个注解的,返回指定支持的注解类集合。
@Override
public Set<String> getSupportedAnnotationTypes() {
Set<String> sets = new LinkedHashSet<String>();
//大部分class而已getName、getCanonicalNam这两个方法没有什么不同的。
//但是对于array或内部类等就不一样了。
//getName返回的是[[Ljava.lang.String之类的表现形式,
//getCanonicalName返回的就是跟我们声明类似的形式。
sets(BindView.class.getCanonicalName());
return sets;
}
//指定Java版本,一般返回最新版本即可
@Override
public SourceVersion getSupportedSourceVersion() {
return SourceVersion.latestSupported();
}
}
动态代理是什么????
自定义事件处理器,继承InvocationHandler ,对实现的invoke方法进行代理设置 ,在主函数中调Porxy.newProxyInstance( ),返回值就是代理对象。