注解提高篇:自定义注解处理器(APT)

摘要:
0x01继承AbstractProcessor抽象类定义注释注释后,我们需要一个注释处理器来处理我们的自定义注释。ExtendesTypeElement˃注释,RoundEnvironmentroundEnv){returnfalse;}}0x02重写核心方法process()。从上面,我们可以看到process()是扫描、解析和处理注释的核心方法。编写一个简单的WondertwoProcessor来提取自定义注释@CustomizeInterface,然后在JavaPool/***自定义注释处理器的帮助下生成一个Java接口文件,将类中的公共方法提取为接口方法*{*Exec:apt factory annotation 3。WondertwoFactory*ProvinceDefiner。java-s./annotaion3*}*创建于2016年10月18日。*/classWondertwoProcessorextendAbstractProcessor{privateProcessingEnvironmentenvir;publicWondertwo处理器{this.env;}@OverridepublicSet<String>getSupportedOptions(){returnsuper.getSupportedOptions();}@OverridepublicSet<String>getSupportedAnnotationTypes(){returnsuper.getSupportedAnnuationTypes;}@Overridepublicsynchronizedvoidint{super.init;}@Overridepublicbooleanprocess(Set˂?

注解提高篇:自定义注解处理器(APT)第1张

0x01 继承AbstractProcessor抽象类


当定义好Annotation注解后,接下来就需要一个注解处理器来处理我们的自定义注解了。实现Java Annotation一般需要继承AbstractProcessor抽象类,并且重写其四个方法来实现提取,解析并处理自定义注解的逻辑如下:

class WondertwoProcessor extends AbstractProcessor {
    //返回注解处理器可处理的注解操作
    @Override
    public Set<String> getSupportedOptions() {
        return super.getSupportedOptions();
    }
    //得到注解处理器可以支持的注解类型
    @Override
    public Set<String> getSupportedAnnotationTypes() {
        return super.getSupportedAnnotationTypes();
    }
    //执行一些初始化逻辑
    @Override
    public synchronized void init(ProcessingEnvironment processingEnv) {
        super.init(processingEnv);
    }
    //核心方法,扫描,解析并处理自定义注解,生成***.java文件
    @Override
    public boolean process(Set<? extends TypeElement> annotations, RoundEnvironment roundEnv) {
        return false;
    }
}

0x02 重写核心方法process()


由上可知process()方法才是扫描,解析,处理注解的核心方法,动手实战一下写一个简单的WondertwoProcessor来提取自定义注解@CustomizeInterface,然后借助JavaPoet生成Java接口文件。

/**
 * 自定义注解处理器,将类中public方法提取为接口方法(不含static方法)
 * {
 *     Exec: apt -factory annotation3.WondertwoFactory
 *     ProvinceDefiner.java -s ../annotaion3
 * }
 * Created by wondertwo on 2016/10/18.
 */
class WondertwoProcessor extends AbstractProcessor {
    private ProcessingEnvironment envir;

    public WondertwoProcessor(ProcessingEnvironment env) {
        this.envir = env;
    }

    @Override
    public Set<String> getSupportedOptions() {
        return super.getSupportedOptions();
    }

    @Override
    public Set<String> getSupportedAnnotationTypes() {
        return super.getSupportedAnnotationTypes();
    }

    @Override
    public synchronized void init(ProcessingEnvironment processingEnv) {
        super.init(processingEnv);
    }

    @Override
    public boolean process(Set<? extends TypeElement> annotations, RoundEnvironment roundEnv) {
        for (TypeElement typeEle : annotations) {
            WondertwoInterface wondertwoInterface = typeEle.getAnnotation(WondertwoInterface.class);
            if (wondertwoInterface == null) break;

            Class clazz = typeEle.getClass();
            if (clazz.getDeclaredMethods().length > 0) {
                try {
                    if (typeEle.getModifiers().contains(Modifier.PUBLIC)
                            && !typeEle.getModifiers().contains(Modifier.STATIC)) {
                        PrintWriter writer = (PrintWriter) envir.getFiler()
                                .createSourceFile(wondertwoInterface.value());
                        writer.println("package " + clazz.getPackage().getName() + ";");
                        writer.println("public interface " + wondertwoInterface.value() + " {");
                        for (Method method : clazz.getDeclaredMethods()) {
                            writer.print("    public ");
                            writer.print(method.getReturnType() + " ");
                            writer.print(method.getName() + " (");
                            int i = 0;
                            for (TypeParameterElement parameter : typeEle.getTypeParameters()) {
                                writer.print(parameter.asType() + " " + parameter.getSimpleName());
                                if (++i < typeEle.getTypeParameters().size())
                                    writer.print(", ");
                            }
                            writer.println(");");
                        }
                        writer.println("}");
                        writer.close();
                    }
                } catch (IOException e) {
                    throw new RuntimeException(e);
                }
            }
        }
        return true;
    }
}

看过《Java编程思想》的同学肯定对上面的实例非常眼熟,书中对应的实例也是提取非静态公有方法生成接口源文件,但由于是JDK6.0标准已经有很多API发生了很大的变化,本例基于JDK8!

可以看到我们只在process()方法中加入了处理注解,生成.java文件的逻辑,这里是的逻辑是根据自定义注解提取对应类的非静态public方法,然后将抽取的非静态共有方法拼接成对应的接口!

0x03 实例探究:Android依赖注解库ButterKnife


不会偷懒的程序员不是一个好程序员,Android开发者对ButterKnife依赖注解库一定耳熟能详,当我们UI布局中控件很多的时候ButterKnife无疑显著提高了开发效率。

作为一个注解库其实现的原理依然是Java Annotation的方式,我们在Github翻出ButterKnife源码文件,找到其核心类——注解处理类ButterKnifeProcessor.java,源码较长删减后如下:

public final class ButterKnifeProcessor extends AbstractProcessor {
  @Override public synchronized void init(ProcessingEnvironment env) {
    super.init(env);
    elementUtils = env.getElementUtils();
    typeUtils = env.getTypeUtils();
    filer = env.getFiler();
  }
  @Override public Set<String> getSupportedAnnotationTypes() {
    Set<String> types = new LinkedHashSet<String>();
    types.add(Bind.class.getCanonicalName());
    for (Class<? extends Annotation> listener : LISTENERS) {
      types.add(listener.getCanonicalName());
    }
    types.add(BindBool.class.getCanonicalName());
    types.add(BindColor.class.getCanonicalName());
    types.add(BindDimen.class.getCanonicalName());
    types.add(BindDrawable.class.getCanonicalName());
    types.add(BindInt.class.getCanonicalName());
    types.add(BindString.class.getCanonicalName());
    return types;
  }
  @Override public boolean process(Set<? extends TypeElement> elements, RoundEnvironment env) {
    Map<TypeElement, BindingClass> targetClassMap = findAndParseTargets(env);
    for (Map.Entry<TypeElement, BindingClass> entry : targetClassMap.entrySet()) {
      TypeElement typeElement = entry.getKey();
      BindingClass bindingClass = entry.getValue();
      try {
        JavaFileObject jfo = filer.createSourceFile(bindingClass.getFqcn(), typeElement);
        Writer writer = jfo.openWriter();
        writer.write(bindingClass.brewJava());
        writer.flush();
        writer.close();
      } catch (IOException e) {
        error(typeElement, "Unable to write view binder for type %s: %s", typeElement,
            e.getMessage());
      }
    }
    return true;
  }
  @Override public Set<String> getSupportedOptions() {
    return Collections.singleton(OPTION_SDK_INT);
  }
}

如果想要进一步了解ButteKnife扫描,解析,处理注解,生成Java代码的每一部细节,可以参考文章:浅析ButterKnife

免责声明:文章转载自《注解提高篇:自定义注解处理器(APT)》仅用于学习参考。如对内容有疑问,请及时联系本站处理。

上篇UILabel[mybatis]list的foreach的使用下篇

宿迁高防,2C2G15M,22元/月;香港BGP,2C5G5M,25元/月 雨云优惠码:MjYwNzM=

相关文章

ubuntu 常用 apt更新命令

常用的APT命令参数: apt-cache search package 搜索包 apt-cache show package 获取包的相关信息,如说明、大小、版本等 sudo apt-get install package 安装包 sudo apt-get install package - - reinstall 重新安装包 sudo apt-get...

Ubuntu 18.04升级matplotlib 3.5

背景 需要使用到matplotlib 3.5的新特性决定升级搜索pip源发现python 3.6最高支持matplotlib 3.2而matplotlib 3.5仅适配python 3.7+ 安装 一定不要尝试卸载18.04自带python 3.6否则系统会崩溃 sudo apt install python3.8 依赖 安装3.8包管理工具 sudo...

Docker打包python flask服务

1、将宿主机上python环境保存到requirements.txt pip3 freeze >requirements.txt 2、新建sources.list文件(apt的源文件) sources.list具体内容如下: $ vi sources.list deb-src http://archive.ubuntu.com/ubuntu x...

debian基本操作

debian基本操作 1 给账户添加sudo权限 1、Debian默认没有sudo功能,因此需要自己安装:# apt-get install sudo2、# chmod +w /etc/sudoers3、# vim /etc/sudoers,添加如下行 root ALL=(ALL:ALL) ALL toney ALL=(ALL:ALL) AL...

Win10下子系统Unbuntu18.04安装nginx

1、Nginx的软件包在Ubuntu默认软件仓库中可用。 安装非常简单,只需键入以下命令: sudo apt update sudo apt install nginx 2、安装完成后,检查Nginx版本: nginx –v  3、启动nginx服务: service nginx start 4、访问 localhost:80 最后补充一句,安装默认的...

ubuntu14.04-安装flash

最近打开搜狐等在线视频,都提示我说flash没有安装,然后就点击安装。进入安装界面 如下所示 然后我们选择版本。之前我一直都是选择版本APT,因为里面标注的是适用于ubuntu10.04+,这时候它会让我们选择用什么软件打开,然后就进入到了 软件管理中心,接着你就会发现,它提示不存在相关资源 然后我们就只能选择手动安装了。这时候,我们可以选择其他计算机需...