`
braveCS
  • 浏览: 72542 次
  • 性别: Icon_minigender_1
  • 来自: 厦门
社区版块
存档分类
最新评论

了解Javac编译器

    博客分类:
  • JVM
阅读更多

本文属笔记,《深入分析JavaWeb》许令波,第4章,仅记录看得懂的部分。(这本书内容很多都在IBM上发表过)。如果有侵权请告诉博主,会马上删除。

 

 

Javac的工作流程:

源码——词法分析器——Token——语法分析器——语法树——语义分析器——注解语法树——代码生成器——字节码

 

1)词法分析

读取源代码,一个字节一个字节地读进来,找到这些字节中哪些是定义的语法关键词,如Java中的ifelseforwhile等关键词,要识别哪些if是合法的关键词、哪些不是。

从源码中找到一些规范化的Token流,就像人类语言中,给你一句话要能分辨出哪些是一个词语、哪些是标点符号、哪些是动词、哪些是名词等。

       Scanner负责具体读取和归类不同词法的操作,判断哪些字符组合是一个TokenJavacParser规定了哪些词是符合Java语言规范规定的词:package语法、import语法、类定义、field定义、method定义、变量定义、表达式定义等,每个语法表达式用分号结束。

2)语法分析

Token流进行语法分析,检查这些关键词组合在一起是不是符合Java语言规范,如if的后面是不是紧跟着一个布尔判断表达式。就像人类语言中,是不是有主谓宾,主谓宾结合得是不是正确,语法是不是正确。

形成一个符合Java语言规范的抽象语法树,抽象语法树是一个结构化的语法表达式形式,它的作用是把语言的主要词法用一个结构化的形式组织在一起。这颗语法树可以被我们后面按照新的规则重新组织。

       token流组建成更加结构化的语法树,也就是间一个个单词组装成一句话,一个完整的语句。哪些词语组合在一起是主语、哪些是谓语、哪些是宾语、哪些是定语,要做进一步区分。Java语法树使得Java源代码更加结构化:每个语法树上的节点都是com.sun.tools.javac.tree.JCTree的一个实例,①每个语法节点都会实现一个接口xxxTree,这个接口又继承于com.sun.source.tree.Tree接口,如IfTree语法节点表示一个if类型的表达式,BinaryTree语法节点代表一个二元操作表达式等;②每个语法节点都是com.sum.tools.javac.tree.JCTree的子类,并且会实现xxxTree接口类,这个类的类名类似于JCxxx,如实现IfTree接口的实现类为JCIf,实现BinaryTree接口的类为JCBinary等;③所有的JCxxx类都作为一个静态内部类在JCTree类中。

 

3)语义分析

       把一些难懂的、复杂的语法转化成更加简单的语法。这个步骤类似将难懂的文言文转化成大家都能懂的白话文或者注解一下一些成语,便于人们更好地理解。

       将复杂的语法转化成最简单的语法,对应到Java中,如将foreach转成for循环结构,还有注解等,最后形成一个注解过后的抽象语法树,这颗语法树更接近目标语言的语法规则。

com.sum.tools.javac.comp.Enter:符号表的构建

1)将Java类中的符号输入到符号表中

1)给类添加默认的构造函数

com.sun.tools.javac.processing.JavacProcessingEnvironmentannotation处理

2)处理annotation注解

com.sun.tools.javac.comp.Attr:标注和语法检查

3)检查操作变量类型是否匹配,操作数|方法返回值类型匹配com.sun.tools.javac.comp.Check

3)检查变量、方法或类的访问是否合法、变量是否是静态变量、变量在使用前是否已经初始化com.sun.tools.javac.comp.Resolve

3)推导出泛型方法中的参数类型com.sum.tools.javac.comp.Infer

3)将一些常量进行合并处理com.sum.tools.javac.comp.ConstFold

com.sun.tools.javac.comp.Flow数据流分析

4)检查变量在使用前是否已经正确赋值

4)包装final修饰的变量不会被重新赋值

4)方法的返回值类型都要确定

4)检查所有的操作是否可达

4)检查checked exception异常是否已经捕获或抛出

5)解除Java的语法糖

5)去掉无用的代码,如永假的if代码块

5)变量的自动转换,如将int自动包装成Integer类型或者相反的操作等;

 

 

4)代码生成器

       根据经过注解的抽象语法树生成字节码,将一个数据结构转化为另一个数据结构,类似将所有的中文语句翻译成英文单词后按照英文语法组装成英文语句。

       com.sun.tools.javac.jvm.Gen

①将Java方法中的代码块转成符合JVM语法的命令形式,JVM的操作都是基于栈的,所有的操作都必须经过出栈和进栈来完成。

②安装JVM的文件组织格式将字节码输出到以class为扩展名的文件中。

2个类:

Items:任何可寻址的操作项,这些包括本地变量、类实例变量或者常量池中用户自定义的常量等,这些操作项都可以作为一个单位出现在操作栈上

Code:存储生成的字节码和提供一些能够映射操作码的方法

 

Javac中访问者模式的实现

词法分析、语法分析、语义分析和代码生成都要多次遍历语法树,但每次遍历这颗语法树都会进行不同的处理动作。Javac采用访问者模式设计,每次遍历都是一次访问者的执行过程。

访问者模式可以将数据结构和对数据结构的操作解耦,使得增加对数据结构的操作不需要去修改数据结构,也不必去修改原有的操作,而执行时再定义新的Visitor实现者就可以。在Javac中不同的编译阶段都定义了不同的访问者模式实现。

TreeScannerEnterAttrGenFlow等都是作为具体访问者角色,每个访问者都定义了自己的访问规则.

EJCIfJCTryJCBreakJCReturn都是具体节点元素,作为一个稳定的数据结构存在。

 

 

 

 

 

分享到:
评论

相关推荐

    javatools源码-java-bytecode-generator:通过Flex&Bison软件工具构建的类似于javac的编译器。它接收

    (类似于javac的编译器) 概述 Java字节码生成器是在著名的工具Flex&Bison之上构建的编译器,用于接收任何Java源代码(当前仅支持Java Lang的子集)并发出其等效的字节码。 这是PLT开发(P AGC软件大号anguage牛逼...

    Java初学者怎么入门.md一份给Java初学者的入门指南

    3. 解释了Java编译和运行的过程,javac编译器和java虚拟机的作用。 4. 总结了Java编程语言的优点:跨平台、面向对象、高效、安全等。 5. 给出了10条学习Java的建议,如选择好的教材、掌握基本语法、多实践编程、注重...

    Java注释@interface的用法

    java用 @interface Annotation{ } 定义一个注解 @Annotation,一个注解是一个类。...JAVAC编译器,开发工具和其他程序可以用反射来了解你的类以及各种元素上有无任何标记,看你有什么标记,就去干相应的事。

    CustomSSM:模仿SSM框架手写自己的框架,加深对框架的理解

    因为在java反射中我使用了1.8独有的一些技术在利用反射进行读取方法信息时,无法得到方法参数的准确名字,而这个反射技术是在1.8才提供的而我们必须配置javac编译器的参数 -parameter 这时才会在.class文件中保存...

    Java程序设计基础:使用命令行方式开发Java程序.pptx

    使用命令行方式开发Java...可执行文件javac.exe位于Java安装目录的bin子目录中,称为Java编译器 (Java Compiler——javac名称的由来),用于对指定的Java源代码进行编译。 打开一个DOS窗口,进入Welcome.java文件所在

    JDK(Java Development Kit)是Java开发工具包,提供了Java开发所需的工具和库

    编译器:JDK包含了Java编译器(javac),可以将Java源代码编译成字节码文件(.class文件)。 JVM(Java虚拟机):JDK包含了Java虚拟机,可以在不同平台上运行Java程序,实现了Java的“一次编写,到处运行”的特性。...

    Java虚拟机

    10.2 Javac编译器 10.2.1 Javac的源码与调试 10.2.2 解析与填充符号表 10.2.3 注解处理器 10.2.4 语义分析与字节码生成 10.3 Java语法糖的味道 10.3.1 泛型与类型擦除 10.3.2 自动装箱、拆箱与遍历循环 ...

    matlab判断为整数代码-sortingComparison:在Scala,Clojure和Java中实现的不同排序算法,以及时间和样式的比

    应该注意的是,在执行此测试之前,我对Java的了解比对Scala或Clojure的了解要多,因此应该花很多精力来考虑编译器错误。 尽管Clojure的一些错误可能表明更轻松的编程体验。 #####为每种语言编写每种算法期间的编译/...

    Java经典入门教程pdf完整版

    Javac Hello.java Netwo Hello class Interpreter code Runtime generator/ Hardware 1:编写代码 首先把我们想要计算机做的事情,通过Java表达出来,写成Java文件,这个过程就是 编写代码的过程。如上图所示的 Hello ...

    JAVA面试题最全集

    简单介绍您所了解的structs。 1.xml在项目中的作用 2.s-EJB 与 e-EJB的区别 3.会话面的作用 4.cmp与bmp的优缺点 5.j2me程序的必需的几个部分 6.c/s与b/s的区别 7.构建一个connect pool,然后再调用它, 8...

Global site tag (gtag.js) - Google Analytics