对于任何一门语言,要想达到精通的水平,研究它的执行原理(或者叫底层机制)不失为一种良好的方式。在本篇文章中,将重点研究java源代码的执行原理,即从程
序员编写JAVA源代码,到最终形成产品,在整个过程中,都经历了什么?每一步又是怎么执行的?执行原理又是什么?.....
当然,本篇文章的粒度可能稍微侧重于宏观方面,更细粒度的技术分析,需要在接下来的该系列文章中与大家分享....
一 编写java源程序
java源文件:指存储java源码的文件;
当前比较主流的JAVA IDE?
(1)Intellij IDEA(首推荐)
(2)Eclipse
先来看看如下代码:
//MyTest被public修饰,故存储该java源码的文件名为MyTest
public class MyTest {
public static void main(String[] args){
System.out.println("Test Java execute process.");
}
}
//由于MyTest被public修饰了,故Class A不能用public修饰
class A{}
//由于MyTest被public修饰了,故Class B不能用public修饰
class B{}
(1)java源文件名就是该源文件中public类的名称
(2)一个java源文件可以包含多个类,但只允许一个类为public
二 编译java源代码
当java源程序编码结束后,就需要编译器编译,安装好jdk后,我们打开jdk目录,有两个.exe文件,
即javac.exe(编译源代码,即.java文件)和java.exe(执行字节码,即.class文件)
1.切换到MyTest.java文件夹
2.javac.exe编译MyTest.java
编译后,发现e:\Blogs 目录多了以class为后缀的文件:A.class,B.class和MyTest.class
Tip:当javac.exe编译java源代码时,java源代码有几个类,就会编译成一个对应的字节码文件(.class文件),
其中,字节码文件的文件名就是每个类的类名。需要注意的是,类即使不在源文件中定义,但被源文件引用,
编译后,也会编程相应的字节码文件,如类A引用类C,但类C不定义在类A的源文件中,编译后,类C也被编
译成对应的字节码文件C.class
三 执行java源文件
执行java源文件,用java.exe执行即可
到现在,java源程序基本执行结果,并正确打印我们期望的结果,那么,如上的步骤,我们可以总结如下:
如上总结,已经抽象化了在JVM中的执行,接下来,我们将分析,字节码文件(.calss文件)如何在虚拟机中一步一执行的。
四 JVM如何执行字节码文件
(一) 装载字节码文件
当.java源码被javac.exe编译器编译成.class字节码文件后,接下来的工作就交给JVM处理,JVM首先通过类加载器(ClassLoader)
将class文件和相关Java API加载装入JVM,以供JVM后续处理。
在该阶段中,涉及到如下一些基本概念和知识。
1.JDK,JRE和JVM关系
(1)JDK(Java Development Kit),Java开发工具包,主要用于开发,在JDK7前,JDK包括JRE
(2)JRE(Java Runtime Environment),Java程序运行的核心环境,包括JVM和一些核心库
(3)JVM(Java Virtual Machine),VM是一种用于计算设备的规范,它是一个虚构出来的计算机,是通过在实际的计算机上仿真模拟
各种计算机功能来实现的,是JRE核心模块。
2.JVM
JVM是一种用于计算设备的规范,它是一个虚构出来的计算机,是通过在实际的计算机上仿真模拟各种计算机功能来实现的。Java虚拟机
虚拟机的主要任务是装载class文件并执行其中的字节码,不同的Java虚拟机中,执行引擎可能由不同的实现,大致有如下几种引擎:
一次性解释字节码引擎
即时编译引擎
自适应优化器
关于虚拟机的实现方式,采用软件方式、硬件方式和软件硬件结合方式,这个要根据具体厂商而定。
3.什么是ClassLoader
虚拟机的主要任务是装载class文件并执行其中的字节码,而class文件是由虚拟机的类加载器(ClassLoader)完成的,在一个Java虚拟机,
有可能存在多个类加载器。
任何java运用程序,可能会使用两种类加载器,即启动类加载器(bootstrap)和用户自定义类加载器。
启动类加载器是Java虚拟机唯一实现的一部分,它又可分为原始类装载器,系统类装载器或默认类装载器,它的主要作用是从操作系统的
磁盘装载相应的类,如Java API类等。
用户自定义装载类,按照用户自定义的方式来装载类。
(二)将字节码文件存储在JVM内存区
当JAVA虚拟机运行一个程序时,它需要内存来存储许多东西,如字节码,从已装载的class文件中得到的其他信息,程序创建的对象,传递给
方法的参数,返回值,局部变量以及运算的中间结果等,这些相关信息被组织到“运行时数据区”。
根据厂商的不同,在Java虚拟机中,运行时数据区也有所不同,有些运行时数据区由线程共享,有些只能由某个特定线程共享。运行时数据区
大致可分几个区:方法区,堆区,栈区,PC寄存器区和本地方法栈区。
在该阶段中,涉及到如下基本概念和知识。
1.方法区
方法区用来存储解析被加载的class文件的相关信息。当虚拟装载一个class文件后,它会从这个class文件包含的二进制数据中解析类型信息,然后将
该相关信息存储到方法区中。
2.堆
堆是用来存储相关引用类型的,如new对象。当程序运行时,虚拟机会把所有该程序在运行时创建的对象都放到堆中。
3.PC寄存器
PC寄存器主要用来存储线程。当新创建一个线程时,该线程都将得到一个自己的PC寄存器(程序计数器)以及一个java栈。
Java虚拟机没有寄存器,其指令集使用Java栈来存储中间数据。
4.栈区
栈区主要用来存储值类型的,如基本数据类型,需要注意的时,String为引用类型,是存在堆中的。Java栈是由许多栈
帧组成的,一个栈帧包含一个Java方法调用的状态,当线程调用一个方法时,虚拟机压入一个新的栈帧到该线程的Java栈中
,当该方法返回时,这个栈帧从Java栈中弹出。
(三)执行引擎与运行时数据区交互
运行时数据区为执行引擎提供了执行环境和相关数据,执行引擎通过与运行时数据区交互,从而获取
执行时需要的相关信息,存储执行的中间结果等
(四)执行引擎与本地方法接口
当要执行本地方法时,执行引擎将调用本地方法接口来获取相关OS本地方法,需要注意的是,本地方法与操作系统强耦合的。
(五)JVM在具体操作系统上执行
JVM通过调用本地接口来获取本地方法,从而实现在具体的平台上执行,如在Linux系统上执行,在Window系统上
执行和在Unix系统上执行。
写好Java代码的30条经验总结(转)
成为一个优秀的Java程序员,有着良好的代码编写习惯是必不可少的.下面就让我们来看看代码编写的30条建议吧. (1) 类名首字母应该大写.字段.方法以及对象(句柄)的首字母应小写.对于所有标识符,其中 ...
写好Java代码的30条经验总结
成为一个优秀的Java程序员,有着良好的代码编写习惯是必不可少的.下面就让我们来看看代码编写的30条建议吧. (1) 类名首字母应该大写.字段.方法以及对象(句柄)的首字母应小写.对于所有标识符,其中 ...
阿里大佬教你,如何写好 Java 代码!
点击上方蓝色链接,关注并"设为星标" Java干货,每天及时推送 阿里大佬分享的一篇很不错的文章,推荐收藏! 导读 明代王阳明先生在谈为学之道时说: 私欲日生 ...
java代码求分数等级的输出~~~
总结:无论是switch-case-break语句 都不要忘了跳出当前循环,即break; 还有这个输入的分数我如何控制在100以内???? package com.c2; //实现分数等级的输出. ...
写出优质Java代码的4个技巧(转)
http://geek.csdn.net/news/detail/238243 原文:4 More Techniques for Writing Better Java 作者:Justin Alban ...
在java代码中显示json字符串(怎么避免json字符串中双引号在java代码中显示)
String log = "eyJvcmRlckluZm8iOnsiaWQiOjEwNzQwNCwib3JkZXJJZCI6MjczNjQyMSwicHJvZHVjdENvZGUiOjQ1N ...
Android视频通话Java代码
近期学习Android平台下跨平台音视频通信开发,尽管网上有非常多开源项目供我们參考学习.但音视频效果非常一般,还有非常多不稳定的因素,毕竟是开源嘛.在国内我找到了一个比較好音视频通信解决方式(百 ...
恕我直言,在座的各位根本写不好Java!
其实,本不想把标题写的那么恐怖,只是发现很多人干了几年 Java 以后,都自认为是一个不错的 Java 程序员了,可以拿着上万的工资都处宣扬自己了,写这篇文章的目的并不是嘲讽和我一样做 Java 的同 ...
如何写出让java虚拟机发生内存溢出异常OutOfMemoryError的代码
程序小白在写代码的过程中,经常会不经意间写出发生内存溢出异常的代码.很多时候这类异常如何产生的都傻傻弄不清楚,如果能故意写出让jvm发生内存溢出的代码,有时候看来也并非一件容易的事.最近通过学习< ...
随机推荐
zjuoj 3780 Paint the Grid Again
http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemCode=3780 Paint the Grid Again Time Limit: 2 ...
.NET跨平台实践:用C#开发Linux守护进程(转)
Linux守护进程(Daemon)是Linux的后台服务进程,它脱离了与控制终端的关联,直接由Linux init进程管理其生命周期,即使你关闭了控制台,daemon也能在后台正常工作. 一句话,为L ...
从C#到Objective-C,循序渐进学习苹果开发(1)--准备开发账号和开发环境
本随笔系列主要介绍从一个Windows平台从事C#开发到Mac平台苹果开发的一系列感想和体验历程,本系列文章是在起步阶段逐步积累的,希望带给大家更好,更真实的转换历程体验,因为一旦方方面面都精通了,也 ...
Mono for android真难用
最近要写个Android小项目,真的很少,几个按钮发送tcp或udp而已. 作为.net程序员当前是c#当先,Mono for android真是好,直接用c#一切都是那么熟悉,但发布时问题来了,需几 ...
Oracle EBS-SQL (SYS-13):查询DBA在系统中的打Patch的信息.SQL
查询DBA在系统中的打补丁信息 1. select * from ad_patch_drivers 2. select * from ad_pat ...
echarts x轴或y轴文本字体颜色改变
1:x轴文本字体颜色改变 xAxis : [ { type : 'category', data : ['<30','30-','40-','50-','60-','>=70'], axi ...
Android中Socket通信之TCP与UDP传输原理
一.Socket通信简介 Android与服务器的通信方式主要有两种,一是Http通信,一是Socket通信.两者的最大差异在于,http连接使用的是"请求-响应方式",即在请求时 ...
jmeter jsr223脚本引用变量的问题
发现jmeter的一个问题不知道算不算bug. 具体表现为,在脚本中通过"${varName}"的方式引用前面使用vars.put("varName",&quo ...
此请求已被阻止,因为当用在 GET 请求中时,会将敏感信息透漏给第三方网站。若要允许 GET 请求,请将 JsonRequestBehavior 设置为 AllowGet。
1.问题描述 mvc从一个路径获取所有的图片信息,ajax方法如下: function getimages(day) { var year = $("#selYear").val( ...
MVC中子页面如何引用模板页中的jquery脚本
MVC中子页面如何引用模板页中的jquery脚本 最近在学习mvc,遇到了一个问题:在html页面中写js代码,都是引用mvc5自带的jquery脚本,虽然一拖(将指定的jquery脚本如 jquer ...