编译时注解-基础篇

在Java中,注解不仅可以提高开发者效率,也可以让程序变得“面向接口编程”。目前,很多框架都用到了这个技术,如SpringMVC,Mybatis,Butterknife等。什么是注解请大家自行百度,今天我要说的是编译时注解的基础知识(仅仅是基础,我也只是做个记录)。至于为什么要说编译时注解呢,因为他运行时效率高。当然运行时注解也很有用,如著名的网络框架Retrofit用的就是运行时注解。

下面我们一般看代码,一边了解:

// com.johan.util.ioc.annotation.InjectView为要解析的注解类名,支持通配符*,如com.johan.util.ioc.annotation.*
@SupportedAnnotationTypes("com.example.util.ioc.annotation.InjectView") 
// 注明java版本
@SupportedSourceVersion(SourceVersion.RELEASE_6)
// 继承AbstractProcessor,如果没有找到,请添加对应版本jar包
public class ViewInjectProcessor extends AbstractProcessor {
    
    // Element辅佐工具
    private Elements elementUtils;

    @Override
    public synchronized void init(ProcessingEnvironment env) {
        super.init(env);
        elementUtils = env.getElementUtils();
    }

    /**
     * 解析注解
     **/
    @Override
    public boolean process(Set annotations, RoundEnvironment roundEnv) {

        String fullClassName, className, packageName;

        // InjectView.class为我的注解类
        for (Element ele : roundEnv.getElementsAnnotatedWith(InjectView.class)) {
             // 打印element信息
             processingEnv.getMessager().printMessage(Kind.NOTE, "ele = " + ele);
             // 如果解析的是类的注解
             if (ele.getKind() == ElementKind.CLASS) {
                  // 类的Element 
                  TypeElement classElement = (TypeElement) ele;
                  // 包的Element,也就是类的封装Element
                  PackageElement packageElement = (PackageElement) ele.getEnclosingElement();
                  // 类的全名:com.example.util.MainActivity
                  fullClassName= classElement.getQualifiedName().toString();
                  // 类名:MainActivity
                  className = classElement.getSimpleName().toString();
                  // 包名:com.example.util
                  packageName = packageElement.getQualifiedName().toString();
                  // 获取类的注解值
                  int layoutId = classElement.getAnnotation(InjectView.class).value();
                  // 打印信息   
                  processingEnv.getMessager().printMessage(Kind.NOTE, "annatated class : packageName = " + packageName + " , className = " + className + " , fqClassName = " + fqClassName);
             } else if (ele.getKind() == ElementKind.FIELD) {
                  // 属性的Element
                  VariableElement varElement = (VariableElement) ele;
                  // 类的Element,就是属性的封装Element
                  TypeElement classElement = (TypeElement) ele.getEnclosingElement();
                  // 类的全名:com.example.util.MainActivity
                  fullClassName= classElement.getQualifiedName().toString();
                  // 包的Element
                  PackageElement packageElement = elementUtils.getPackageOf(classElement);
                  // 包名:com.example.util
                  packageName = packageElement.getQualifiedName().toString();
                  // 类名 :MainActivity
                  className = classElement.getSimpleName().toString();
                  // 获取属性的注解
                  int id = varElement.getAnnotation(InjectView.class).value();
                  String fieldName = varElement.getSimpleName().toString();
                  String fieldType = varElement.asType().toString();
                  // 打印信息                    
                  processingEnv.getMessager().printMessage(Kind.NOTE, "annatated field : fieldName = " + varElement.getSimpleName().toString() + " , id = " + id + " , fileType = " + fieldType);   
             }
             /**
              * 下面可以用JavaFileObject jfo = processingEnv.getFiler().createSourceFile方法生成JavaFileObject对象,然后Writer writer = jfo.openWriter()方法得到一个writer对象,最后java代码文字,动态生成的java文件
              **/
             return true;
        }
    }

}        

大概就这样了。如果有不对的地方,希望指出,谢谢!

你可能感兴趣的:(编译时注解-基础篇)