今天给大家分享的是Java编译器API简介,文章部分内容摘自【优锐课】学习笔记。

Java编译器API

Java编译器API是Java模块(称为java.compiler)的一部分。该模块包括语言模型和注释处理,以及编译器API。它定义了Java编程语言和编译器工具的类型和模型声明,可以在执行期间从应用程序代码中调用它们。注释处理有助于访问注释处理器,可以将其视为Java编译器的插件。它使注释处理器和注释处理工具环境之间能够通信。模型,元素和类型包处理Java编程语言的元素,而util包则帮助处理程序元素和类型。

编译工具

javax.tools包提供了与Java编译器一起使用的接口和类,并且可以在执行期间从程序中调用它。 它提供了一个框架,该框架允许客户端从其自己的应用程序代码定位和运行编译器。它还提供了服务提供者接口(SPI),用于对诊断的结构化访问和用于覆盖文件访问的文件抽象。ToolProvider类提供了编译器API的入口点。此类提供了一些方法来定位编译器的工具提供者。 例如,我们可以轻松地找到系统中安装的编译器支持的Java源版本列表。

 JavaCompiler compiler = ToolProvider.getSystemJavaCompiler();
for(SourceVersion sv:compiler.getSourceVersions()){
System.out.println(sv);
}

输出如下(根据系统中安装的版本)。

 RELEASE_3
RELEASE_4
RELEASE_5
RELEASE_6
RELEASE_7
RELEASE_8
RELEASE_9
RELEASE_10
RELEASE_11

在这种情况下,ToolProvider会找到默认的编译器。通过使用服务提供者机制,还可以找到替代的编译器或工具。如果某些供应商提供Java编译器,则jar文件将包含文件META-INF / service / javax.tool.JavaCompiler,并且将包含一行:com.vendor.VendorJavaCompiler。我们可以将jar文件放入类路径中,并按以下方式定位它:

 JavaCompiler vendorJavaCompiler =
ServiceLoader.load(JavaCompiler.class).iterator().next();

ServiceProvider是Java的util类之一,用于查找和加载部署在执行环境中的服务提供者。

找到JavaCompiler后,就可以通过Java源代码执行各种编译诊断任务。为了说明这个想法,让我们首先创建一个简单的类,如下所示:

 package com.mano.jcapidemo;
import java.util.Random;
public class MyClass {
public static void main(String[] args){
Random r = new Random();
System.out.println("Today your Lucky Number is:
"+r.nextInt(10));
}
}

现在,在创建Java源文件之后,我们可以使用名为DiagnosticCollector的诊断收集器类将诊断收集在列表中。

创建另一个类,从该类中我们将调用编译器来编译上述类MyClass,并将诊断信息报告给该类。换句话说,我们将创建一个应用程序来加载Java源文件,并由Java编译器对其进行编译,并且,如果源代码中有任何错误,请确保将其报告给主机应用程序。

 package com.mano.jcapidemo;

 import com.mano.annotation.CustomAnnotation;
import java.util.Set;
import javax.annotation.processing.AbstractProcessor;
import javax.annotation.processing.RoundEnvironment;
import javax.annotation.processing.SupportedAnnotationTypes;
import javax.annotation.processing.SupportedSourceVersion;
import javax.lang.model.SourceVersion;
import javax.lang.model.element.Element;
import javax.lang.model.element.ElementKind;
import javax.lang.model.element.TypeElement;
import javax.tools.Diagnostic;
@SupportedAnnotationTypes("com.mano.annotation.CustomAnnotation")
@SupportedSourceVersion(SourceVersion.RELEASE_10)
public class CustomAnnotationProcessor extends
AbstractProcessor {
public CustomAnnotationProcessor() {
} public Boolean process(Set<? extends
TypeElement> annotations,
RoundEnvironment roundEnv) {
for (Element e : roundEnv.getElementsAnnotatedWith
(CustomAnnotation.class)) {
if (e.getKind() != ElementKind.FIELD) {
processingEnv.getMessager().printMessage(
Diagnostic.Kind.WARNING,
"Not a field", e);
continue;
}
}
return true;
}
}

编译器依赖于两种服务:诊断侦听器和文件管理器。如果提供了侦听器,则将诊断信息提供给侦听器;否则,将向侦听器提供诊断信息。否则,诊断将以未指定的格式格式化,并定向到默认的错误输出系统(System.err)。默认情况下,编译器工具与标准文件管理器关联,并且可以与满足其要求的任何其他文件管理器一起正常工作。

注释处理器

编译过程还包括注释处理器。它执行编译由注释驱动的代码的附加过程。处理过程按一系列轮次进行,其中每个轮次处理其上一轮产生的注释子集。实现注释过程的接口是javax.annotation.processin.Processor。实现类必须提供一个无参数的构造函数,以供工具实例化处理器。处理基础结构应遵循某些协议,例如:

  • 通过使用处理器类的无参数构造函数实例化注释处理器。
  • 工具通过传递适当的ProcessingEnvironmentinstance实例来调用init方法。
  • 这些工具调用由Processor接口定义的方法,例如getSupportedAnnotationTypes()getSupportedOptions(), 和getSupportedSourceVersion()。这些方法在每次运行中调用一次,而不是在每个回合中调用一次。
  • 最后,调用Processor对象上的process ()方法。

例如,简单的注释可以定义如下:

 package com.mano.jcapidemo;
import java.lang.annotation.ElementType;
import java.lang.annotation.Target;
@Target(ElementType.FIELD)
public@interface CustomAnnotation {
}

一个非常简单的注释处理器,用于警告将注释应用于字段以外的任何其他元素,如下所示:

 package com.mano.jcapidemo;

 import com.mano.annotation.CustomAnnotation;
import java.util.Set;
import javax.annotation.processing.AbstractProcessor;
import javax.annotation.processing.RoundEnvironment;
import javax.annotation.processing.SupportedAnnotationTypes;
import javax.annotation.processing.SupportedSourceVersion;
import javax.lang.model.SourceVersion;
import javax.lang.model.element.Element;
import javax.lang.model.element.ElementKind;
import javax.lang.model.element.TypeElement;
import javax.tools.Diagnostic;
@SupportedAnnotationTypes("com.mano.annotation.CustomAnnotation")
@SupportedSourceVersion(SourceVersion.RELEASE_10)
public class CustomAnnotationProcessor extends
AbstractProcessor {
public CustomAnnotationProcessor() {
} public Boolean process(Set<? extends
TypeElement> annotations,
RoundEnvironment roundEnv) {
for (Element e : roundEnv.getElementsAnnotatedWith
(CustomAnnotation.class)) {
if (e.getKind() != ElementKind.FIELD) {
processingEnv.getMessager().printMessage(
Diagnostic.Kind.WARNING,
"Not a field", e);
continue;
}
}
return true;
}
}

SupportedAnnotationTypes定义注释处理器将处理哪种类型的注释,SupportedSourceVersion定义其支持的版本。 我们首先扩展AbstractProcessor抽象类,该类允许我们覆盖处理方法。 处理方法内部编写的逻辑完成了所有技巧,这些技巧涉及我们选择设置哪些标准来处理注释。 这最终决定了注释的含义。

元素扫描仪

元素扫描器在编译过程中对所有语言元素执行分析。它根据访问者模式构建,以根据源版本的发布情况,以默认行为扫描程序元素。例如,ElementScanner9根据源版本RELEASE_9和RELEASE_10进行扫描,而ElementScanner8分别根据源版本RELEASE_8进行扫描。这两个类都可以在javax.lang.model.utilpackage中找到。

编译树API Compiler Tree API

有时,有必要将整个Java源文件解析为抽象语法树,尤其是为了进行更深入的分析。Java编译器树API遵守该要求,并与javax.lang.model包紧密关联。它以与元素扫描器相同的模式构建,并且以类似的方式工作。密钥类称为TreePathScanner。它访问所有子树节点,并有助于维护到父节点的路径。要访问特定节点,我们可以简单地覆盖相应的visitorXYZ方法。

总结

Java编译器API从Java应用程序中提供对Java编译器的编程访问。显而易见,此API有更深层的含义,在这里我们只涉及了其中的内容。但是,此快速介绍可能会提供有关在开始使用Java Compiler API时要查找的内容的线索。

参考

Java API文档

Java编译器API简介的更多相关文章

  1. jxl(Java Excel API) 使用方法 【2】

    JAVA EXCEL API简介 Java Excel是一开放源码项目,通过它Java开发人员可以读取Excel文件的内容.创建新的Excel文件.更新已经存在的Excel文件.使用该 API非Win ...

  2. Java SE 6 新特性: 编译器 API

    新 API 功能简介 JDK 6 提供了在运行时调用编译器的 API,后面我们将假设把此 API 应用在 JSP 技术中.在传统的 JSP 技术中,服务器处理 JSP 通常需要进行下面 6 个步骤: ...

  3. Java Content Repository API 简介 转自(https://www.ibm.com/developerworks/cn/java/j-jcr/)

    Java Content Repository API 简介 1 如果曾经试过开发内容管理应用程序,那么您应当非常清楚在实现内容系统时所遇到的固有难题.这个领地有点支离破碎,许多供应商都有自己的私有仓 ...

  4. Java的脚本机制、编译器API

    学习 xxl-job 定时任务时了解到基于 JVM 的 Grovvy 脚本语言.搭建 Jenkins 时知道了编译API 1. Java 脚本机制 Java 的脚本 API 可以让我们调用 JavaS ...

  5. 构建简单的 C++ 服务组件,第 1 部分: 服务组件体系结构 C++ API 简介

    构建简单的 C++ 服务组件,第 1 部分: 服务组件体系结构 C++ API 简介 熟悉将用于 Apache Tuscany SCA for C++ 的 API.您将通过本文了解该 API 的主要组 ...

  6. Java 2D API - 1. 基本概念

    Java 2D API扩展AWT包,对二维图形.文本及成像功能提供了支持,可用于开发复杂的界面.绘图软件和图像编辑器.Java 2D对象位于用户坐标空间(User coordinate space), ...

  7. ZooKeeper系列4:ZooKeeper API简介及编程

    问题导读: 1.ZooKeeper API 共包含几个包? 2.如何使用ZooKeeper API 创建zookeeper应用程序? 1)ZooKeeper API 简介   ZooKeeper AP ...

  8. Java Servlet 技术简介

    Java Servlet 技术简介 Java 开发人员兼培训师 Roy Miller 将我们现有的 servlet 介绍资料修改成了这篇易于学习的实用教程.Roy 将介绍并解释 servlet 是什么 ...

  9. 【Java】Java Servlet 技术简介

    Java 开发人员兼培训师 Roy Miller 将我们现有的 servlet 介绍资料修改成了这篇易于学习的实用教程.Roy 将介绍并解释 servlet 是什么,它们是如何工作的,如何使用它们来创 ...

随机推荐

  1. Python 面向对象-上篇

    概述 面向过程:根据业务逻辑从上到下写垒代码 函数式:将某功能代码封装到函数中,日后便无需重复编写,仅调用函数即可 面向对象:对函数进行分类和封装,让开发“更快更好更强...” 面向过程编程最易被初学 ...

  2. nyoj 55-懒省事的小明(priority_queue)

    55-懒省事的小明 内存限制:64MB 时间限制:3000ms Special Judge: No accepted:8 submit:62 题目描述:       小明很想吃果子,正好果园果子熟了. ...

  3. static declaration follows non-static declaration

    前段时间工作中要为android编译跨平台的第三方库,遇到了arc4random有关函数的“static declaration follows non-static declaration”问题,那 ...

  4. Java程序性能优化之性能概述

    性能的基本概念 一).什么叫程序的性能? 程序运行所需的内存和时间. 二).性能的表现形式: 1).执行速度: 程序的反应是否迅速,响应时间是否足够短. 2).启动时间:程序从运行到可以处理正常业务所 ...

  5. 小白学 Python 爬虫(4):前置准备(三)Docker基础入门

    人生苦短,我用 Python 前文传送门: 小白学 Python 爬虫(1):开篇 小白学 Python 爬虫(2):前置准备(一)基本类库的安装 小白学 Python 爬虫(3):前置准备(二)Li ...

  6. mybatis精讲(三)--标签及TypeHandler使用

    目录 话引 XML配置标签 概览 properties 子标签property resource 程序注入 settings 别名 TypeHandler 自定义TypeHandler EnumTyp ...

  7. linux运维与实践

    1.容器云计算节点负载值高,通过top可以看到Load Average:70.1  71.3  70.8,虚拟机有8个cpu: cpu使用率高导致(R状态)? 同时在top中观察一段时间,消耗cpu最 ...

  8. Hadoop简述

    Haddop是什么? Hadoop是一个由Apache基金会所开发的分布式系统基础架构 主要解决,海量数据的存储和海量数据的分析计算问题. Hadoop三大发行版本 Apache版本最原始(最基础)的 ...

  9. SpringBoot系列之集成Thymeleaf用法手册

    目录 1.模板引擎 2.Thymeleaf简介 2.1).Thymeleaf定义 2.2).适用模板 3.重要知识点 3.1).th:text和th:utext 3.2).标准表达式 3.3).Thy ...

  10. 题解——面积(area.cpp)

    题目来源&题面简述: 思路与算法选择: 只有*里面的部分对我们有用,所以可以将 *号外的部分标记一下. 可以用著名的BFS大法实现此过程.(连通块) 连通块模板: #include<bi ...