Javac之Environment
关于Env的源代码如下:
/** A class for environments, instances of which are passed as
* arguments to tree visitors. Environments refer to important ancestors
* of the subtree that's currently visited, such as the enclosing method,
* the enclosing class, or the enclosing toplevel node. They also contain
* a generic component, represented as a type parameter, to carry further
* information specific to individual passes.
*
* 一个类的环境,它的实例将被作为树访问者的参数进行传递。环境涉及到重要的子树的先祖,
* 子树是目前被访问过的。例如封闭的方法、封闭的类或者是封闭的顶层结点。它们也包含了
* 一个一般性的组件,被描述为类型参数,针对个别的传递携带更多的信息
*
*/
public class Env<A> implements Iterable<Env<A>> {
/** The next enclosing env.
*
* 下一个封闭的环境,和作用域是不同的。这个与语法树的组织嵌套是一样的
*/
public Env<A> next;
/** The env enclosing the current class.
*
* 参考文章:
*/
public Env<A> outer;
/** The tree with which this environment is associated.
*/
public JCTree tree;
/**
* The enclosing toplevel tree.封闭的顶层语法树,语法节点类型为JCCompilationUnit
*/
public JCCompilationUnit toplevel;
/** The next enclosing class definition.
*/
public JCClassDecl enclClass;
/** The next enclosing method definition.
*/
public JCMethodDecl enclMethod;
/** A generic field for further information.
*/
public A info;
// ...
}
主要通过三种途径来获取Env对象,如下:
(1)构造函数创建Env对象并返回
代码如下:
/** Create an outermost environment for a given (toplevel)tree,with a given info field.
*/
public Env(JCTree tree, A info) {
this.next = null;
this.outer = null;
this.tree = tree; // 初始化时就赋值
this.toplevel = null;
this.enclClass = null;
this.enclMethod = null;
this.info = info; // 初始化时就赋值
}
调用的地方如下图所示。

除去Env中的dup方法与子类通过super()进行调用后,剩下了两个主要的地方在调用。
(2)调用dupto()方法获取Env对象
/** Duplicate this environment into a given Environment,
* using its tree and info, and copying all other fields.
*/
public Env<A> dupto(Env<A> that) {
that.next = this;
that.outer = this.outer;
that.toplevel = this.toplevel;
that.enclClass = this.enclClass;
that.enclMethod = this.enclMethod;
return that;
}
被调用的地方如下图所示。
(3)调用dup()方法获取Env对象
/** Duplicate this environment, updating with given tree,
* and copying all other fields.
*/
public Env<A> dup(JCTree tree) { // 更新tree的信息
return dup(tree, this.info);
}
/** Duplicate this environment, updating with given tree and info,
* and copying all other fields.
*/
public Env<A> dup(JCTree tree, A info) { // 更新tree与info的信息
return dupto(new Env<A>(tree, info));
}
如上两个dup()方法都将调用dupto()方法来获取Env对象。
第一个方法的调用地方如下图所示。

第二个方法的调用地方如下图所示。

下面来介绍各个属性。
1、info属性:
参考文章:https://www.cnblogs.com/extjs4/p/9285113.html
2、outer属性:
class A {
static class B { // 只有在toplevel类中可用static,所有的内部类不可以使用static关键字
int b = 2;
public void methodB(){
class C {
Object c = new Object(){
int c = b;
};
{
class D{
public void t() {
int c = b;
}
}
}
}
}
}
}
对于匿名内部类的int c = b语句来说,其outer是JCNewClass(JCNewClass属于类C)。类C的outer为类B的JCBlock。类B的outer为类A。
对于方法t中的int c = b语句来说,语句属于类D,类D的外层为JCBlock(JCBlock{classD{...}},而JCBlock(JCBlock{classD{...}}的outer为methodB的Block。
3、enclClass属性:
一个编译单元JCCompilationUnit的Env下的enclClass属性为:
/** Create a fresh environment for toplevels.
* @param tree The toplevel tree.
*/
public Env<AttrContext> topLevelEnv(JCCompilationUnit tree) {
// 初始化namedImportScope属性
tree.namedImportScope = new ImportScope(tree.packge);
tree.starImportScope = new StarImportScope(tree.packge);
Env<AttrContext> localEnv = new Env<AttrContext>(tree, new AttrContext());
localEnv.toplevel = tree;
// predefClassDef:A dummy class to serve as enclosingClass
// for toplevel environments.
localEnv.enclClass = predefClassDef;
// JCCompilationUnit的环境Scope中使用的是namedImportScope
localEnv.info.scope = tree.namedImportScope;
localEnv.info.lint = lint;
return localEnv;
}
是一个预先定义好的predefClassDef。
public Env<AttrContext> classEnv(JCClassDecl tree, Env<AttrContext> env) {
Scope sp = new Scope(tree.sym);
AttrContext ac = env.info.dup(sp);
Env<AttrContext> localEnv = env.dup(tree, ac);
localEnv.enclClass = tree;
localEnv.outer = env;
localEnv.info.isSelfCall = false;
localEnv.info.lint = null; // leave this to be filled in by Attr,
// when annotations have been processed
return localEnv;
}
被调用的截图如下:

JCClassDecl语法节点的Env的属性enclClass为自己本身,而JCTypeParameter语法节点举例如下:
interface IA{}
interface IB{}
class CA{}
class CB{}
class TestC<T extends CA&IA&IB>{
public <B extends CB&IA&IB> void method(B b){}
}
则visitTypeParameter()方法的cd属性类型为JCClassDecl,分别为:
1、public abstract class T extends CA implements IA,IB{} 最终生成的Env如下截图。

2、public abstract class B extends CB implements IA,IB{} 最终生成的Environment如下截图。

对于Scope属性举个例子,如下:
package com.test07;
import java.io.FileInputStream;
import java.io.InputStream;
class TestScope {
public int a = 2;
public void test(int b) {
int c = 3;
for (int d = 3; a < 6; a++) {
int e = 4;
}
}
}
Scope如下截图。

在如上的例子中,JCVariableDecl节点及子结点的staticLevel都为1,而其父结节点及以上为0。再举个例子,如下:
public class Test3{
static{ // staticLevel = 1
// 在static块中不允许再有static出现,如
// static int a = 1;
// static class A{}
// class A{
// public static void m1(){}
// }
int a = 1; // staticLevel = 1
}
// 这个类的staticLevel为0,与一般类没有任何区别
static class Inner{ // staticLevel = 0
public static void m1(){ // staticLevel = 1
int b = 1; // staticLevel = 1
}
}
}
举个例子,如下:
interface I{
void m1();
}
public class Test3{
static void t(){}
static I a = new I(){
@Override
public void m1() {
t();
}
};
}
那么t()的环境Env的结构如下截图。
这是outer属性,而next属性和语法树的组织结构是严格一致的,如下截图。

Javac之Environment的更多相关文章
- Fedora javac 命令提示 [javac: 未找到命令...]
[joy@localhost ~]$ java -version openjdk version "1.8.0_91" OpenJDK Runtime Environment (b ...
- Java SE series:1. environment configure and Hello world! [We use compiler and packager to create an application!]
1. cli (command line interface) and gui (graphic user interface) use javahome path, search classpath ...
- Storm(1) - Setting Up Development Environment
Setting up your development environment 1. download j2se 6 SDK from http://www.oracle.com/technetwor ...
- JDK Environment Variable And Change default JDK
Environment Variable : change(import) /etc/bashrc export JAVA_HOME=/software/jdk1.8.0 export PATH=$J ...
- Config the Android 5.0 Build Environment
In this document Choosing a Branch Setting up a Linux build environment Installing the JDK ...
- 在XP系统下搭建maven环境出的问题 Unable to locate the Javac Compiler in: C:\Program Files\Java\jre6\..\lib\tools.jar
Build errors for spider; org.apache.maven.lifecycle.LifecycleExecutionException: Failed to execute g ...
- google官方提供的编译android源码的环境初始化,Initializing a Build Environment
原文网址:http://source.android.com/source/initializing.html Initializing a Build Environment IN THIS DOC ...
- 解决linux下javac -version和java -version版本显示不一致
解决linux下javac -version和java -version版本显示不一致 [javascript] view plaincopy [root@localhost usr]# $JAVA_ ...
- 第六章-Javac符号表
需要参考: (1)Architecture of a Java Compiler (2)关于符号Symbol第一篇 (3)关于符号Symbol第二篇 (4)关于类型Type (5)关于作用域范围Sco ...
随机推荐
- 怎样运用servlet
制作登陆界面 login.html <!DOCTYPE html> <html> <head> <title>login.html</title& ...
- CAS实战の自定义注销
步骤一 在cas server端,设置/WebContent/WEB-INF/cas-servlet.xml: <bean id="logoutAction" class=& ...
- MSP430 G2553 低功耗模式LPMx
MSP430除了正常运行时的active模式外,还支持五种低功耗模式(Low-power mode),分别为LPM0.LPM1.LPM2.LPM3.LPM4,由状态寄存器中的CPUOFF.OSCOFF ...
- 疑难杂症--SQL SERVER 18056的错误
朋友遇到一个很棘手的问题,查看服务器日志,报以下错误: ::,spid296,未知,错误: ,严重性: ,状态: . ::,spid495,未知, The client was unable < ...
- asp.net 导出 Excel 身份证格式显示格式问题
<%# Eval("数据").ToString()+" " %> 加上 Excel 中 不会显示科学计数法
- npm 安装less
npm install less less-loader --save 在style加上lang="less" 就可以直接用了
- c# BindingSource 类
1.引言 BindingSource组件是数据源和控件间的一座桥,同时提供了大量的API和Event供我们使用.使用这些API我们可以将Code与各种具体类型数据源进行解耦:使用这些Eve ...
- Java 使用json 做配置文件
概述 经常会用到通过配置文件,去配置一些参数,java里面本来是有配置文件的,但是导入很麻烦的,自从我用了json之后,从此一切配置文件都见鬼去吧. 1.下载gson解析json文件的jar包 ...
- BZOJ1558 等差数列
题目链接:戳我 实话实话,看了几篇题解真的没看懂,我觉得讲的都有问题.这里对于线段树维护的s写了一点我自己的理解. 看到等差数列,我们考虑对数列做差,这样如果是等差数列,那么值应该相等.(比较容易维护 ...
- 前端基础--css基本语法,选择器
一.css概述 CSS(Cascading Style Sheet)层叠样式表,定义如何显示HTML元素,给HTML设置样式,让它更加美观.当浏览器读到一个样式表,它就会按照这个样式表来对文档进行格式 ...