javac 概述

javac 是jdk bin目录下的一个脚本。 用于编译 java程序的源代码,但是 其实现的本质 是基于 jdk 标准类库中的 javac类库实现,所以java的编译器实质上是一个 java程序。

javac脚本 仅是一个便于启动以及传递参数的脚本文件,其内部依旧运行了 java程序。

javac 又被称作前端编译器,仅负责 源代码 与 字节码之间的转换,而在jvm内部 还存在 一个后置编译器,根据热点探测技术 可以将最有价值的 字节码转换为 机器码执行从而提升java程序的运行效率。

javac 的意义就在于 将源码编译为字节码,同时 做一些 词法,语法,语义上的检查,最后生成可供jvm运行的字节码文件。

javac 源码

在 lib 中的 tools jar 包中 sun.tools.javac;  包下管理者 java前端编译器 的class文件。 Main 类 中的 main 方法的执行 是javac程序的执行入口。

 public static void main(String args[])
{
//将标准错误流获取
PrintStream printstream = System.err;
if (Boolean.getBoolean("javac.pipe.output"))
printstream = System.out;
//创建 编译器对象
Main main1 = new Main(printstream, "javac");
//调用编译器的 compile方法进行 编译 并接受 args 为参数,该参数就是 javac 后面携带的参数
System.exit(main1.compile(args) ? 0 : main1.exitStatus);
}

compile方法的 编译过程 概括性的分析:

1.解析与填充符号表

解析: 对java源代码的字节流进行读取解析,进行两个大致的步骤,词法解析以及语法解析

词法解析: 识别 java源码中存在的表达语义的逻辑单元,列如 关键字 变量名 参数名 每一个逻辑单元 称为 标量。

语法解析:将各个独立的 标量按照java语法规范形成 java语法数,语法树的每一个节点代表一个操作,运算,或者 方法调用。

填充符号表: 解析后的语法树最顶级的节点将被用来填充在符号表中,符号表存储着各个语法树的最顶级节点,填充后的符号表最终形成 待处理表。

符号表就是一个遵从java语法的结构规范,用于组织语法树的逻辑顺序。

2.插入式注解处理器

jdk1.5后引入注解功能,注解是一种应用字节码 属性中类的元数据进行操作的一种编程机制。

处理表形成后 会自动检测是否有注解器需要执行,若有则执行注解处理器。注解处理器实现了在可插入式的编译期改变编译过程的功能。

其本质就是 再次修改 处理表中的语法树。 一旦语法树被修改,则将再次进行 词法,语法分析并填充符号表的过程。

3.语义分析并生成字节码

语义分析: 再次对语法树中的节点进行校验。对数据类型以及控制逻辑进行检测。

标量检测: 检验关键字是否使用正确,类型转换是否正确等。

数据与控制流分析: 对控制流程的逻辑进行校验。

语法糖解析:  编程语言为了 增加代码的可读性,以及减少编程出错率,提供了一些并不影响程序运行期仅在编译期有效的编程机制。

java语言中语法糖 有 泛型,拆箱与装箱,foreach循环,可变参数,switch,枚举等,在编译期将转换为字节码遵守的规范形式。

泛型使用类型擦出,拆装箱调用了valueOf与xxValue方法,foreach是迭代器 可变参数是数组,switch本质是 if else 的嵌套。

字节码替换:  在生成类的字节码之时,编译器后做一些默认性质的操作,当没有显示声明的构造器,则会创建默认的无参构造器,构造器分为 实例构造器与类构造器

在字节码层面 类构造器 是指多个static代码块中的语句 收敛生成的<cinit>指令。而构造代码块与显示的构造器将收敛生成实例构造器。

同时还会将 String类型的 +与+= 操作,默认替换为 对 StringBuffer或 StrignBudiuer的操作。

最后生成字节码。

代码如下:

 public synchronized boolean compile(String as[])
{
String s = null;
String s1 = null;
String s2 = null;
String s3 = null;
boolean flag = false;
String s4 = null;
short word0 = 45;
short word1 = 3;
File file = null;
File file1 = null;
String s5 = "-Xjcov";
String s6 = "-Xjcov:file=";
int i = 0x41004;
long l = System.currentTimeMillis();
Vector vector = new Vector();
boolean flag1 = false;
Object obj = null;
String s7 = null;
String s8 = null;
String s9 = null;
exitStatus = 0;
try
{
as = CommandLine.parse(as);
}
catch (FileNotFoundException filenotfoundexception)
{
error("javac.err.cant.read", filenotfoundexception.getMessage());
System.exit(1);
}
catch (IOException ioexception)
{
ioexception.printStackTrace();
System.exit(1);
}
label0:
for (int j = 0; j < as.length; j++)
{
if (as[j].equals("-g"))
{
if (s8 != null && !s8.equals("-g"))
error("main.conflicting.options", s8, "-g");
s8 = "-g";
i |= 0x1000;
i |= 0x2000;
i |= 0x40000;
continue;
}
if (as[j].equals("-g:none"))
{
if (s8 != null && !s8.equals("-g:none"))
error("main.conflicting.options", s8, "-g:none");
s8 = "-g:none";
i &= 0xffffefff;
i &= 0xffffdfff;
i &= 0xfffbffff;
continue;
}
if (as[j].startsWith("-g:"))
{
if (s8 != null && !s8.equals(as[j]))
error("main.conflicting.options", s8, as[j]);
s8 = as[j];
String s10 = as[j].substring("-g:".length());
i &= 0xffffefff;
i &= 0xffffdfff;
i &= 0xfffbffff;
do
{
do
{
if (s10.startsWith("lines"))
{
i |= 0x1000;
s10 = s10.substring("lines".length());
} else
if (s10.startsWith("vars"))
{
i |= 0x2000;
s10 = s10.substring("vars".length());
} else
if (s10.startsWith("source"))
{
i |= 0x40000;
s10 = s10.substring("source".length());
} else
{
error("main.bad.debug.option", as[j]);
usage_error();
return false;
}
if (s10.length() == 0)
continue label0;
} while (!s10.startsWith(","));
s10 = s10.substring(",".length());
} while (true);
}
if (as[j].equals("-O"))
{
if (s9 != null && !s9.equals("-O"))
error("main.conflicting.options", s9, "-O");
s9 = "-O";
continue;
}
if (as[j].equals("-nowarn"))
{
i &= -5;
continue;
}
if (as[j].equals("-deprecation"))
{
i |= 0x200;
continue;
}
if (as[j].equals("-verbose"))
{
i |= 1;
continue;
}
if (as[j].equals("-nowrite"))
{
flag1 = true;
continue;
}
if (as[j].equals("-classpath"))
{
if (j + 1 < as.length)
{
if (s1 != null)
error("main.option.already.seen", "-classpath");
s1 = as[++j];
} else
{
error("main.option.requires.argument", "-classpath");
usage_error();
return false;
}
continue;
}
if (as[j].equals("-sourcepath"))
{
if (j + 1 < as.length)
{
if (s != null)
error("main.option.already.seen", "-sourcepath");
s = as[++j];
} else
{
error("main.option.requires.argument", "-sourcepath");
usage_error();
return false;
}
continue;
}
if (as[j].equals("-sysclasspath"))
{
if (j + 1 < as.length)
{
if (s2 != null)
error("main.option.already.seen", "-sysclasspath");
s2 = as[++j];
} else
{
error("main.option.requires.argument", "-sysclasspath");
usage_error();
return false;
}
continue;
}
if (as[j].equals("-bootclasspath"))
{
if (j + 1 < as.length)
{
if (s2 != null)
error("main.option.already.seen", "-bootclasspath");
s2 = as[++j];
} else
{
error("main.option.requires.argument", "-bootclasspath");
usage_error();
return false;
}
continue;
}
if (as[j].equals("-extdirs"))
{
if (j + 1 < as.length)
{
if (s3 != null)
error("main.option.already.seen", "-extdirs");
s3 = as[++j];
} else
{
error("main.option.requires.argument", "-extdirs");
usage_error();
return false;
}
continue;
}
if (as[j].equals("-encoding"))
{
if (j + 1 < as.length)
{
if (s7 != null)
error("main.option.already.seen", "-encoding");
s7 = as[++j];
} else
{
error("main.option.requires.argument", "-encoding");
usage_error();
return false;
}
continue;
}
if (as[j].equals("-target"))
{
if (j + 1 < as.length)
{
if (s4 != null)
error("main.option.already.seen", "-target");
s4 = as[++j];
int k = 0;
do
{
if (k >= releases.length)
break;
if (releases[k].equals(s4))
{
word0 = majorVersions[k];
word1 = minorVersions[k];
break;
}
k++;
} while (true);
if (k == releases.length)
{
error("main.unknown.release", s4);
usage_error();
return false;
}
} else
{
error("main.option.requires.argument", "-target");
usage_error();
return false;
}
continue;
}
if (as[j].equals("-d"))
{
if (j + 1 < as.length)
{
if (file != null)
error("main.option.already.seen", "-d");
file = new File(as[++j]);
if (!file.exists())
{
error("main.no.such.directory", file.getPath());
usage_error();
return false;
}
} else
{
error("main.option.requires.argument", "-d");
usage_error();
return false;
}
continue;
}
if (as[j].equals(s5))
{
i |= 0x40;
i &= 0xffffbfff;
i &= 0xffff7fff;
continue;
}
if (as[j].startsWith(s6) && as[j].length() > s6.length())
{
file1 = new File(as[j].substring(s6.length()));
i &= 0xffffbfff;
i &= 0xffff7fff;
i |= 0x40;
i |= 0x80;
continue;
}
if (as[j].equals("-XO"))
{
if (s9 != null && !s9.equals("-XO"))
error("main.conflicting.options", s9, "-XO");
s9 = "-XO";
i |= 0x4000;
continue;
}
if (as[j].equals("-Xinterclass"))
{
if (s9 != null && !s9.equals("-Xinterclass"))
error("main.conflicting.options", s9, "-Xinterclass");
s9 = "-Xinterclass";
i |= 0x4000;
i |= 0x8000;
i |= 0x20;
continue;
}
if (as[j].equals("-Xdepend"))
{
i |= 0x20;
continue;
}
if (as[j].equals("-Xdebug"))
{
i |= 2;
continue;
}
if (as[j].equals("-xdepend") || as[j].equals("-Xjws"))
{
i |= 0x400;
if (out == System.err)
out = System.out;
continue;
}
if (as[j].equals("-Xstrictdefault"))
{
i |= 0x20000;
continue;
}
if (as[j].equals("-Xverbosepath"))
{
flag = true;
continue;
}
if (as[j].equals("-Xstdout"))
{
out = System.out;
continue;
}
if (as[j].equals("-X"))
{
error("main.unsupported.usage");
return false;
}
if (as[j].equals("-Xversion1.2"))
{
i |= 0x800;
continue;
}
if (as[j].endsWith(".java"))
{
vector.addElement(as[j]);
} else
{
error("main.no.such.option", as[j]);
usage_error();
return false;
}
} if (vector.size() == 0 || exitStatus == 2)
{
usage_error();
return false;
}
BatchEnvironment batchenvironment = BatchEnvironment.create(out, s, s1, s2, s3);
if (flag)
output(getText("main.path.msg", batchenvironment.sourcePath.toString(), batchenvironment.binaryPath.toString()));
batchenvironment.flags |= i;
batchenvironment.majorVersion = word0;
batchenvironment.minorVersion = word1;
batchenvironment.covFile = file1;
batchenvironment.setCharacterEncoding(s7);
String s11 = getText("main.no.memory");
String s12 = getText("main.stack.overflow");
batchenvironment.error(0L, "warn.class.is.deprecated", "sun.tools.javac.Main");
try
{
for (Enumeration enumeration = vector.elements(); enumeration.hasMoreElements();)
{
File file2 = new File((String)enumeration.nextElement());
try
{
batchenvironment.parseFile(new ClassFile(file2));
}
catch (FileNotFoundException filenotfoundexception1)
{
batchenvironment.error(0L, "cant.read", file2.getPath());
exitStatus = 2;
}
} Object obj1 = batchenvironment.getClasses();
do
{
if (!((Enumeration) (obj1)).hasMoreElements())
break;
ClassDeclaration classdeclaration = (ClassDeclaration)((Enumeration) (obj1)).nextElement();
if (classdeclaration.getStatus() == 4 && !classdeclaration.getClassDefinition().isLocal())
try
{
classdeclaration.getClassDefinition(batchenvironment);
}
catch (ClassNotFound classnotfound) { }
} while (true);
obj1 = new ByteArrayOutputStream(4096);
boolean flag2;
do
{
flag2 = true;
batchenvironment.flushErrors();
Enumeration enumeration1 = batchenvironment.getClasses();
do
{
if (!enumeration1.hasMoreElements())
break;
ClassDeclaration classdeclaration1 = (ClassDeclaration)enumeration1.nextElement();
switch (classdeclaration1.getStatus())
{
case 1: // '\001'
case 2: // '\002'
default:
break; case 0: // '\0'
if (!batchenvironment.dependencies())
continue;
// fall through case 3: // '\003'
batchenvironment.dtEvent((new StringBuilder()).append("Main.compile (SOURCE): loading, ").append(classdeclaration1).toString());
flag2 = false;
batchenvironment.loadDefinition(classdeclaration1);
if (classdeclaration1.getStatus() != 4)
{
batchenvironment.dtEvent((new StringBuilder()).append("Main.compile (SOURCE): not parsed, ").append(classdeclaration1).toString());
continue;
}
// fall through case 4: // '\004'
if (classdeclaration1.getClassDefinition().isInsideLocal())
{
batchenvironment.dtEvent((new StringBuilder()).append("Main.compile (PARSED): skipping local class, ").append(classdeclaration1).toString());
continue;
}
flag2 = false;
batchenvironment.dtEvent((new StringBuilder()).append("Main.compile (PARSED): checking, ").append(classdeclaration1).toString());
SourceClass sourceclass = (SourceClass)classdeclaration1.getClassDefinition(batchenvironment);
sourceclass.check(batchenvironment);
classdeclaration1.setDefinition(sourceclass, 5);
// fall through case 5: // '\005'
SourceClass sourceclass1 = (SourceClass)classdeclaration1.getClassDefinition(batchenvironment);
if (sourceclass1.getError())
{
batchenvironment.dtEvent((new StringBuilder()).append("Main.compile (CHECKED): bailing out on error, ").append(classdeclaration1).toString());
classdeclaration1.setDefinition(sourceclass1, 6);
continue;
}
flag2 = false;
((ByteArrayOutputStream) (obj1)).reset();
batchenvironment.dtEvent((new StringBuilder()).append("Main.compile (CHECKED): compiling, ").append(classdeclaration1).toString());
sourceclass1.compile(((OutputStream) (obj1)));
classdeclaration1.setDefinition(sourceclass1, 6);
sourceclass1.cleanup(batchenvironment);
if (sourceclass1.getNestError() || flag1)
continue;
String s14 = classdeclaration1.getName().getQualifier().toString().replace('.', File.separatorChar);
String s15 = (new StringBuilder()).append(classdeclaration1.getName().getFlatName().toString().replace('.', '$')).append(".class").toString();
File file3;
if (file != null)
{
if (s14.length() > 0)
{
file3 = new File(file, s14);
if (!file3.exists())
file3.mkdirs();
file3 = new File(file3, s15);
} else
{
file3 = new File(file, s15);
}
} else
{
ClassFile classfile = (ClassFile)sourceclass1.getSource();
if (classfile.isZipped())
{
batchenvironment.error(0L, "cant.write", classfile.getPath());
exitStatus = 2;
continue;
}
file3 = new File(classfile.getPath());
file3 = new File(file3.getParent(), s15);
}
try
{
FileOutputStream fileoutputstream = new FileOutputStream(file3.getPath());
((ByteArrayOutputStream) (obj1)).writeTo(fileoutputstream);
fileoutputstream.close();
if (batchenvironment.verbose())
output(getText("main.wrote", file3.getPath()));
}
catch (IOException ioexception1)
{
batchenvironment.error(0L, "cant.write", file3.getPath());
exitStatus = 2;
}
if (batchenvironment.print_dependencies())
sourceclass1.printClassDependencies(batchenvironment);
break;
}
} while (true);
} while (!flag2);
}
catch (OutOfMemoryError outofmemoryerror)
{
batchenvironment.output(s11);
exitStatus = 3;
return false;
}
catch (StackOverflowError stackoverflowerror)
{
batchenvironment.output(s12);
exitStatus = 3;
return false;
}
catch (Error error1)
{
if (batchenvironment.nerrors == 0 || batchenvironment.dump())
{
error1.printStackTrace();
batchenvironment.error(0L, "fatal.error");
exitStatus = 4;
}
}
catch (Exception exception)
{
if (batchenvironment.nerrors == 0 || batchenvironment.dump())
{
exception.printStackTrace();
batchenvironment.error(0L, "fatal.exception");
exitStatus = 4;
}
}
int i1 = batchenvironment.deprecationFiles.size();
if (i1 > 0 && batchenvironment.warnings())
{
int j1 = batchenvironment.ndeprecations;
Object obj2 = batchenvironment.deprecationFiles.elementAt(0);
if (batchenvironment.deprecation())
{
if (i1 > 1)
batchenvironment.error(0L, "warn.note.deprecations", new Integer(i1), new Integer(j1));
else
batchenvironment.error(0L, "warn.note.1deprecation", obj2, new Integer(j1));
} else
if (i1 > 1)
batchenvironment.error(0L, "warn.note.deprecations.silent", new Integer(i1), new Integer(j1));
else
batchenvironment.error(0L, "warn.note.1deprecation.silent", obj2, new Integer(j1));
}
batchenvironment.flushErrors();
batchenvironment.shutdown();
boolean flag3 = true;
if (batchenvironment.nerrors > 0)
{
String s13 = "";
if (batchenvironment.nerrors > 1)
s13 = getText("main.errors", batchenvironment.nerrors);
else
s13 = getText("main.1error");
if (batchenvironment.nwarnings > 0)
if (batchenvironment.nwarnings > 1)
s13 = (new StringBuilder()).append(s13).append(", ").append(getText("main.warnings", batchenvironment.nwarnings)).toString();
else
s13 = (new StringBuilder()).append(s13).append(", ").append(getText("main.1warning")).toString();
output(s13);
if (exitStatus == 0)
exitStatus = 1;
flag3 = false;
} else
if (batchenvironment.nwarnings > 0)
if (batchenvironment.nwarnings > 1)
output(getText("main.warnings", batchenvironment.nwarnings));
else
output(getText("main.1warning"));
if (batchenvironment.covdata())
{
Assembler assembler = new Assembler();
assembler.GenJCov(batchenvironment);
}
if (batchenvironment.verbose())
{
l = System.currentTimeMillis() - l;
output(getText("main.done_in", Long.toString(l)));
}
return flag3;
}

javac 实现原理的更多相关文章

  1. 第二章 Javac编译原理

    注:本文主要记录自<深入分析java web技术内幕>"第四章 javac编译原理" 1.javac作用 将*.java源代码文件转化为*.class文件 2.编译流程 ...

  2. Knowledge Point 20180303 对比编译器、解释器与Javac编译原理

    编译器与Javac编译原理 在前文我们知道了Java是一种编译语言和解释语言,它的源代码经过编译器Javac编译为能够被JVM识别的二进制语言,然后JVM将其解释为能够被平台识别的机器语言.那么什么是 ...

  3. 第四章 Javac编译原理

    4.1 Javac是什么 是一种编译器,将JAVA源代码(.java文件)语言先转化成JVM能够识别的一种语言(.class文件),然后由JVM将JVM语言再转化成当前机器可以识别的机器语言. 4.2 ...

  4. 第四章 Javac编译原理(待续)

    Javac是什么 Javac编译器的基本结构 Javac工作原理分析 设计模式解析之访问者模式

  5. javac编译原理(一)

    我们都知道,计算机只能识别二进制语言,是不能直接识别java c c++等高级语言的.将高级语言转化成计算机可以是别的二进制语言,这个过程就叫编译. 有次面试,面试官问了一道“java的编译原理是什么 ...

  6. Javac 编译原理

    写在前面 JDK & JRE  JRE(Java Runtime Enviroment)是Java的运行环境.面向Java程序的使用者,而不是开发者.如果你仅下载并安装了JRE,那么你的系统只 ...

  7. javac编译原理

    javac编译器的作用就是将符合java语言规范的源代码转化成符合java虚拟机规范的java字节码 经历:词法分析器->语法分析器->语义分析器->编译字节码 四个过程生成字节码文 ...

  8. Javac编译原理 《深入分析java web 技术内幕》第四章

    javac编译的四个主要的流程: 词法分析器:将源码转换为Token流 将源代码划分成一个个Token(找出java语言中的关键字) 语法分析器:将Token流转化为语法树 将上述的一个个Token组 ...

  9. 深入分析Java Web技术内幕(修订版)

    阿里巴巴集团技术丛书 深入分析Java Web技术内幕(修订版)(阿里巴巴集团技术丛书.技术大牛范禹.玉伯.毕玄联合力荐!大型互联网公司开发应用实践!) 许令波 著   ISBN 978-7-121- ...

随机推荐

  1. 使用Hexo+Github一步步搭建属于自己的博客(基础)

    前言:电脑系统为window 10专业版,64位 相关步骤: 1.安装Node.js和配置好Node.js环境,打开cmd命令行,成功界面如下 2.安装Git和配置好Git环境,安装成功的象征就是在电 ...

  2. 容器与Docker简介(二)什么是DOCKER——微软微服务电子书翻译系列

    Docker是一个开源项目,用于将应用程序部署自动化,作为可在云端或本地运行的可移植,自包含的容器. Docker同时也是一家促进和发展这项技术的公司,与云,Linux以及Windows的供应商(包括 ...

  3. TinyOS编程思想和Nesc基础语法

    TinyOS操作系统由nesc语言写成,从程序员角度看,它的基本作用就是提供了一组API接口以及一些编程规则. 具体来说,基于nesc语言的TinyOS编程行为具有以下特点: a.兼容C语言:使用ne ...

  4. ios微信支付 v3

    V2版本和V3版本存在很大的差异. 1. 从成功通过微信支付商户的资料审核返回的邮件开始: 你可以获得这些参数  appid,appSecret,partnerID,    partnerKey(从平 ...

  5. Paint the Grid Reloaded(缩点,DFS+BFS)

    Leo has a grid with N rows and M columns. All cells are painted with either black or white initially ...

  6. Entropy

    Entropy Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others) Total Submi ...

  7. Spring AOP分析(2) -- JdkDynamicAopProxy实现AOP

    上文介绍了代理类是由默认AOP代理工厂DefaultAopProxyFactory中createAopProxy方法产生的.如果代理对象是接口类型,则生成JdkDynamicAopProxy代理:否则 ...

  8. Page visibility 页面可见性

    一直以来,判断页面是不是当前可见标签,浏览器有没有缩小都是比较麻烦的.   通过页面可见性API可以获得相关信息document.hidden  判断页面当前是不是可见的document.visibi ...

  9. 企业级memcached部署(session共享)

    服务端部署 第一个里程碑:安装依赖关系 Memcache用到了libevent这个库用于Socket的处理. [root@nfs01 ~]# yum install libevent libevent ...

  10. HTML学习笔记 CSS文本及字体及连接及列表(a标签使用及缩进) 案例 第七节 (原创)参考使用表

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...