现有如下代码,那么我们的JVM是怎么执行的呢

public class Test{
public static vodi main(String[] args){
MaYun my=new MaYun();
my.work();
}
} public class MaYun{ public void work(){
System.out.print("i am working")
}
}

首先,当我们启动虚拟机进程这个进程首先从classpath中找到Text.class文件,读取这个文件中的二进制数据,然后把Text类的类信息存放到运行时数据区的方法区中。这一过程称为Text类的加载过程。
接着,Java虚拟机定位到方法区中Text类的Main()方法的字节码,开始执行它的指令。这个main()方法的第一条语句就是:
MaYun my=new MaYun();
语句很简单啦,就是让java虚拟机创建一个MaYun 实例,并且呢,使引用变量my引用这个实例。貌似小case一桩哦,就让我们来跟踪一下Java虚拟机,看看它究竟是怎么来执行这个任务的:
1、 Java虚拟机一看,不就是建立一个MaYun 实例吗,简单,于是就直奔方法区而去,先找到Sample类的类型信息再说。结果呢,嘿嘿,没找到, 这会儿的方法区里还没有MaYun 类呢。可Java虚拟机也不是一根筋的笨蛋,于是,它发扬“自己动手,丰衣足食”的作风,立马加载了MaYun 类, 把MaYun 类的类型信息存放在方法区里。
2、 好啦,资料找到了,下面就开始干活啦。Java虚拟机做的第一件事情就是在堆区中为一个新的MaYun 实例分配内存, 这个MaYun 实例持有着指向方法区的MaYun 类的类型信息的引用。这里所说的引用,实际上指的是MaYun 类的类型信息在方法区中的内存地址, 其实,就是有点类似于C语言里的指针啦~~,而这个地址呢,就存放了在MaYun 实例的数据区里。
3、 在JAVA虚拟机进程中,每个线程都会拥有一个方法调用栈,用来跟踪线程运行中一系列的方法调用过程,栈中的每一个元素就被称为栈帧,每当线程调用一个方 法的时候就会向方法栈压入一个新帧。这里的帧用来存储方法的参数、局部变量和运算过程中的临时数据。OK,原理讲完了,就让我们来继续我们的跟踪行动,位 于“=”前的my是一个在main()方法中定义的变量,可见,它是一个局部变量,因此,它被会添加到了执行main()方法的主线程的JAVA方 法调用栈中。而“=”将把这个my变量指向堆区中的MaYun 实例,也就是说,它持有指向MaYun 实例的引用。
OK,到这里为止呢,JAVA虚拟机就完成了这个简单语句的执行任务。参考我们的行动向导图,我们终于初步摸清了JAVA虚拟机的一点点底细了
接下来,JAVA虚拟机将继续执行后续指令,然后依次执行它们的work()方法。当JAVA虚拟机 执行my.work()方法时,JAVA虚拟机根据局部变量my持有的引用,定位到堆区中的MaYun 实例,再根据MaYun 实例持有的引用,定位到方法去中MaYun 类的类型信息,从而获得work()方法的字节码,接着执行work()方法包含的指令。该指令被OS识别后打印到控制台就完成了他的操作了。

类的加载顺序。
(1) 父类静态代码块(包括静态初始化块,静态属性,但不包括静态方法)
(2) 子类静态代码块(包括静态初始化块,静态属性,但不包括静态方法 )
(3) 父类非静态代码块( 包括非静态初始化块,非静态属性 )
(4) 父类构造函数
(5) 子类非静态代码块 ( 包括非静态初始化块,非静态属性 )
(6) 子类构造函数
其中:类中静态块按照声明顺序执行,并且(1)和(2)不需要调用new类实例的时候就执行了(意思就是在类加载到方法区的时候执行的)
2.其次,需要理解子类覆盖父类方法的问题,也就是方法重写实现多态问题。
Base b = new Sub();它为多态的一种表现形式,声明是Base,实现是Sub类, 理解为 b 编译时表现为Base类特性,运行时表现为Sub类特性。
当子类覆盖了父类的方法后,意思是父类的方法已经被重写,题中 父类初始化调用的方法为子类实现的方法,子类实现的方法中调用的baseName为子类中的私有属性。

04-23 18:44