笔记:Javac编译器
Javac编译器是把 *.java 文件转换为 *.class 文件,是一个前端编译器;对应着有一种把字节码转变为机器码的编译器,称为JIT编译器(Just In Time Compiler),比如 HotSpot VM 的C1、C2编译器;把 *.java 文件编译成机器码的编译器称为静态提前编译器;
Javac编译器编译的过程可以为3个过程:
1、解析与填充符号表:
这个过程又可以细分为词法分析、语法分析和填充符号表;
词法分析:词法分析是将源代码的字符流转变为标记(Token)集合,单个字符是程序编写过程的最小元素,而标记则是编译过程的最小元素,关键字、变量名、字面量、运算符都可以成为标记。
语法分析:语法分析是根据Token序列构造抽象语法树的过程,抽象语法树(Abstract Syntax Tree)是一种用来描述程序代码语法结构的树形表示方式,语法树的每一个节点都代表着程序代码中的一个语结构。
符号表(Symbol Tree):符号表是一组符号地址和符号信息构造的表格,符号表登记的信息在编译的不同阶段都要用到。
2、插入式注解处理器的注解处理:插入式注解器相当于编译器的插件,我们可以通过注解处理器读取、修改、添加抽象语法树中的任意元素。如果对抽象语法树进行了修改,编译器将会重新回到解析及填充符号表的过程重新处理,直到所有插入式注解处理器都没有再对语法树进行修改为止。
3、语义分析与字节码生成过程:语法分析的主要任务是对结构上正确的源程序进行上下文有关性质的审核,保证源代码程序符合逻辑。
语义分析又可以细分为 标注检查和数据及控制流分析:
标注检查:标注检查检查的内容包括变量使用前是否被声明、变量与赋值之间的数据类型是否能够匹配,常量折叠(比如 int a = 1 * 2; 会处理为 int a = 3;)等等;
数据及控制流检查:检查内容有程序局部变量在使用前是否有赋值、方法的每一条路径是否都有返回值,是否所有的受查异常都被正常的处理 等等;
语法糖:指在计算机语言中添加某种语法,这种语法对语言的功能并没有影响,但是方便程序员使用;
字节码生成:字节码生成是把前面各个步骤所生成的信息(语法树、符号表)转化成字节码写到磁盘的过程,在这个过程中还进行了不少代码添加和转换的工作,比如生成实例构造器和类构造器;
Java 语法糖
1、泛型与类型擦除:
泛型是JDK 1.5 的一项新增特性,它的本质是参数化类型的应用,也就是说所操作的数据类型被指定为一个参数。这种参数类型可以用在类、接口和方法的创建中,分别称为泛型类、泛型接口和泛型方法。
Java的泛型是一种伪泛型,它只存在于程序的源代码中,在编译后的字节码文件中,就已经替换为原来的原生类型。
2、自动拆箱、装箱和遍历循环:
自动装箱、自动拆箱反编译后对应包装类的包装、还原方法;
遍历循环 反编译后对应 迭代器实现;
变长参数 反编译后 变成了一个数组类型的参数;
// 以下是源代码,包含了泛型,自动装箱,自动拆箱、遍历循环、变长参数共5种语法糖
List<Integer> list = Arrays.asList(1, 2, 3, 4, 5);
int sum = 0;
for (int i : list) {
sum += i;
}
System.out.println(sum); //反编译后的源代码
List list = Arrays.asList(new Integer[] {
null, null, null, null,
(new Integer[5][3] =
(new Integer[5][2] =
(new Integer[5][1] =
(new Integer[5][0] = Integer.valueOf(1)).valueOf(2)).valueOf(3)).valueOf(4)).valueOf(5) });
int i = 0;
for (Iterator iterator = list.iterator(); iterator.hasNext(); ) {
int j = ((Integer)iterator.next()).intValue();
i += j;
}
System.out.println(i);
3、自动装箱的陷阱:
// 源代码
Integer a = 1;
Integer b = 2;
Integer c = 3;
Integer d = 3;
Integer e = 321;
Integer f = 321;
Long g = 3L; System.out.println(c == d);
System.out.println(e == f);
System.out.println(c == (a + b));
System.out.println(c.equals(a + b));
System.out.println(g == (a + b));
System.out.println(g.equals(a + b)); // 反编译后的代码
Integer integer1 = Integer.valueOf(1);
Integer integer2 = Integer.valueOf(2);
Integer integer3 = Integer.valueOf(3);
Integer integer4 = Integer.valueOf(3);
Integer integer5 = Integer.valueOf(321);
Integer integer6 = Integer.valueOf(321);
Long l = Long.valueOf(3L); System.out.println((integer3 == integer4));
System.out.println((integer5 == integer6));
System.out.println((integer3.intValue() == integer1.intValue() + integer2.intValue()));
System.out.println(integer3.equals(Integer.valueOf(integer1.intValue() + integer2.intValue())));
System.out.println((l.longValue() == (integer1.intValue() + integer2.intValue())));
System.out.println(l.equals(Integer.valueOf(integer1.intValue() + integer2.intValue()))); // 运行结果
true
false
true
true
true
false //说明
1. 包装类的 “==” 运算在遇到算术运算符会自动拆箱
2. 包装类的 equals 方法先会判断比较的类型是否 instanceof 包装类
3. 当Integer,Long包装类在自动装箱时如果值在 -128 ~ 127 之间时会共享缓存值
4、条件编译
javac编译器会把条件分支中永远不会执行的分支消除掉。
笔记:Javac编译器的更多相关文章
- java-关于java_home配置,classpath配置和javac,java命令,javac编译器,和java虚拟机之间的关系
在每个人学习java的第一步,都是安装jdk ,jre,配置java_home,classpath,path. 为什么要做这些?在阅读java-core的时候,看到了原理,p141. 一 关于类的共享 ...
- 早期(编译器)优化--javac编译器
java语言的“编译期”其实是一段“不确定”的操作过程,可能是指一个前端编译器把.java变成.class的过程,也可能是指虚拟机的后端运行期编译器(JLT)把字节码转变成机器码的过程,也有可能是使用 ...
- 第一章-Javac编译器介绍
1.Javac概述 编译器可以将编程语言的代码转换为其他形式,如Javac,将Java语言转换为虚拟机能够识别的.class文件形式.而这种将java源代码(以.java做为文件存储格式)转换为cla ...
- Javac编译器详解
学习<深入了解Java虚拟机>有一段时间了,大概理解了Java从源代码编译到执行出结果的过程,也能明确的知道Java是半解释性语言.在执行源代码时,先通过Javac编译器对源代码进行词法分 ...
- 早期javac编译器优化
学习<深入了解Java虚拟机>有一段时间了,大概理解了Java从源代码编译到执行出结果的过程,也能明确的知道Java是半解释性语言.在执行源代码时,先通过Javac编译器对源代码进行词法分 ...
- Javac 编译器
编译过程 Javac 编译过程大致可以分为1个准备过程和3个处理过程: 准备过程:初始化插入式注解处理器. 解析与填充符号表过程,包括: 词法.语法分析,将源代码的字符流转变为标记集合,构造出抽象语法 ...
- 关于Javac编译器的那点事(一)
Javac是什么? 它是一种编译器,将Java对人非常友好的语言,编译转化对所有机器都非常友好的语言,即:JVM能够识别的语言,也就是Java字节码.而Java字节码,说白了就是一连串二进制数字. J ...
- 对openjdk的javac编译器扩展了一个语法糖
我的扩展功能描述如下: 在java的现有语法中加入var来声明变量,并且可以根据初始化数据来自动类型推导. 举两个例子: 例一: 如下JAVA代码(注意这里的var是新语法): import java ...
- JVM系列五(javac 编译器).
一.概述 我们都知道 *.java 文件要首先被编译成 *.class 文件才能被 JVM 认识,这部分的工作主要由 Javac 来完成,类似于 Javac 这样的我们称之为前端编译器: 但是 *.c ...
随机推荐
- poj1930 Dead Fraction
思路: 循环小数化分数,枚举所有可能的循环节,取分母最小的那个. 实现: #include <iostream> #include <cstdio> #include < ...
- SCANF输入错误
while((a<=0||a>=10)||(b<=0||b>=10)) { fflush(stdin); cout<<" ...
- TNS-00511: 无监听程序
这里到服务里面打开 tns 的监听服务
- 从GridView中直接导出数据到Excel文件 处理导出乱码 类型“GridView”的控件“XXXX”必须放在具有 runat=server 的窗体标记内。”的异常
导出到Excel方法: <span style="color: rgb(0, 0, 255);">public</span> <span style= ...
- 提高SQL查询效率 的10大方法
一.查询条件精确,针对有参数传入情况 二.SQL逻辑执行顺序 FROM–>JOIN–>WHERE–>GROUP–>HAVING–>DISTINCT–>ORDER–& ...
- C# 设置系统环境变量
using Microsoft.Win32; using System; using System.Collections.Generic; using System.ComponentModel; ...
- chown - 修改文件所有者和组别
总览 chown [options] user [:group] file... POSIX 选项: [-R] GNU 选项(最短格式): [-cfhvR] [--dereference] [--re ...
- Js获取操作系统版本 && 获得浏览器版本
//利用原生Js获取操作系统版本function getOS() { var sUserAgent = navigator.userAgent; var isWin = (navigator.plat ...
- 11-2 numpy/pandas/matplotlib模块
目录 numpy模块 一维数组 二维数组 列表list和numpy的区别 获取多维数组的行和列 多维数组的索引 高级功能 多维数组的合并 通过函数方法创建多维数组 矩阵的运算 求最大值最小值 nump ...
- JAVA基础——设计模式之单列模式
一:单例设计模式 Singleton是一种创建型模式,指某个类采用Singleton模式,则在这个类被创建后,只可能产生一个实例供外部访问,并且提供一个全局的访问点. 单例设计模式的特点: 单例类只能 ...