文档章节

java 程序是如何运行的?

SEOwhywhy
 SEOwhywhy
发布于 08/16 23:56
字数 3110
阅读 1
收藏 0

  不知大家有没有思考过,当我们使用IDE写了一个Demo类,并执行main函数打印 hello world时都经历了哪些流程么?
  
  想通过这篇文章来分析分析Java的执行流程,或者换句话说想聊聊Java的编译期与运行期的流程。
  
  开门见山
  
  编译期间都做了什么
  
  运行期间都做了什么
  
  1. 开门见山
  
  public class MyApp {
  
  public static void main(String[] args) {
  
  System.out.println("hello world");
  
  假如我们写了一个MyApp.java,并要打印‘hello world’ 那它需要经过哪些步骤?
  
  第一步:compile
  
  通过编译器进行编译,从Java源码 ---> Java 字节码
  
  这个编译器则是jdk 里的javac 编译器,我们只需 javac MyApp.java 即可以编译该源码,javac 编译器位于jdk --> bin -->javac
  
  第二步:load and execute
  
  加载java 字节码并执行
  
  可以通过jdk 里的java命令运行java字节码,我们只需 java MyApp.class 即可加载并执行该字节码,当运行java命令时,JRE将与您指定的类一起加载。然后,执行该类的主要方法。
  
  public class AddHeaderAttribute: ResultFilterAttribute
  
  private readonly string _name;
  
  private readonly string _value;
  
  public AddHeaderAttribute(string name, string value)
  
  _name = name;
  
  _value = value;
  
  public override void OnResultExecuting(www.jujinyulee.com ResultExecutingContext context)
  
  context.HttpContext.Response.Headers.Add(
  
  _name,new string[www.hnawesm.com] { _value });
  
  base.OnResultExecuting(context);
  
  特性允许过滤器接受参数,如下,可将此特性添加到控制器或操作中,并为其指定所需 HTTP 头的名称和值:
  
  [AddHeader("Author", "Ruby Lu"www.gouyiflb.cn]
  
  public class HomeController : Controller
  
  以下几种过滤器接口可以自定义为相应特性的实现:
  
  ActionFilterAttribute
  
  ExceptionFilterAttribute
  
  ResultFilterAttribute
  
  FormatFilterAttribute
  
  ServiceFilterAttribute
  
  TypeFilterAttribute
  
  4.取消和短路
  
  通过设置传入过滤器方法的上下文参数中的 Result 属性,可以在过滤器管道的任意一点短路管道。比如,下面的 ShortCircuitingResourceFilter 将阻止它之后管道内的所有过滤器,包括所有操作过滤器:
  
  复制代码
  
  public class www.dongfangyuld.com ShortCircuitingResourceFilter:Attribute,IResourceFilter
  
  public void OnResourceExecuted(ResourceExecutedContext www.jintianxuesha.com context)
  
  java命令位于jdk --> bin -->java。
  
  上面只是大概讲了运行一个java程序的流程,下面再从编译期以及运行期的角度在剖析一下细节。
  
  2. 编译期间都做了什么?
  
  编译器(compiler)是一种计算机程序,它会将某种编程语言写成的源代码(原始语言)转换成另一种编程语言(目标语言)。
  
  编译期都做了什么?从我们使用者角度看无非就是把源代码编译成了可被虚拟机执行的字节码,但是从平台(编译器)角度看,它所经历的流程还不少。
  
  毕竟总不能给你什么以.java为后缀的文件都进行编译吧,需要有各种校验解析步骤
  
  2.1 解析与填充符号表
  
  词法语法分析
  
  词法分析
  
  是指把源代码的字符流转为标记(Token)集合,标记(Token)是编译阶段的最小单元,字符则是编程阶段源码的最小单元。
  
  比如,int i = 0由4个标记构成分别是「int,i,=,0」编译器只认识这些标记,词法分析过程就是识别一个个标记的过程
  
  语法分析
  
  则是把生成的标记集合 构成一个语法树,每个节点代表程序代码中的语法结构,如包,类型,修饰符,运算符等等。
  
  填充符号表
  
  通过了上面的词义语义分析之后我们需要把数据存起来,以供后续流程使用,编译器会以key-value的形式存储数据,以符号地址为key符号信息为value,具体形式没做限制可以是树状符号表或者有序符号表等。
  
  在语义分析中,根据符号表所登记的内容 语义检查和产生中间代码,在目标代码生成阶段,当对符号表进行地址分配时,该符号表是检查的依据。
  
  2.2 注解处理器
  
  注解与普通的Java代码一样,是在运行期间发挥作用的。我们可以把它看做是一组编译器的插件,在这些插件里面,可以读取、修改、添加抽象语法树中的任意元素。
  
  如果这些插件在处理注解期间对语法树进行了修改,编译器将回到解析及填充符号表的过程重新处理,直到所有插入式注解处理器都没有再对语法树进行修改为止。
  
  换句话说当我们处理注解时如果修改了语法树的话会重新执行分析以及符号填充过程,把注解也填充进来,直到处理完所有注解。
  
  2.3 语义分析
  
  语法分析以及处理注解之后,编译器获得了程序代码的抽象语法树,语法树能表示一个结构正确的源程序的抽象,但无法保证源程序是符合逻辑的。
  
  说白了,语法树上的内容单个来说是合法的但是结合到上下文语义则未必是合法的。
  
  比如定义了两个变量
  
  int a = 1; boolean b = false; int c = a + b
  
  以上 都能构成结构正确的语法树,但是根据语义分析之后编译是通不过,Java语言中是不合乎逻辑的。
  
  2.4 解语法糖
  
  Java 中最常用的语法糖主要有泛型、变长参数、条件编译、自动拆装箱、内部类等。虚拟机并不支持这些语法,它们在编译阶段就被还原回了简单的基础语法结构,这个过程成为解语法糖。
  
  换句话说,不论你是否使用Java的语法糖,最终到jvm哪里的时候都是一样的,jvm不支持语法糖,所以需要编译阶段解语法糖,语法糖的初衷是用来提升开发效率,而不是代码性能。
  
  2.5 字节码生成
  
  字节码生成是Javac编译过程的最后一个阶段,在Javac源码里面由com.sun.tools.javac. jvm.Gen类来完成。字节码生成阶段前面各个步骤所生成的信息(语法树、符号表)转化成字节码写到磁盘中,主要工作就是把语法树和符号表加工成字节码文件。
  
  3. 运行期间都做了什么?
  
  java的运行期主要是处理编译器产生的字节码,包括加载与执行。
  
  3.1 加载器与验证器
  
  java提供类加载器把虚拟机外部的字节码资源载入到虚拟机的运行时环境(主要是指虚拟机的方法区)
  
  并提供字节码验证器来保证载入的字节码是安全合法的,对程序没有危害的。
  
  加载器 (Class Loader)
  
  当字节码还没被类加载器加载之前它目前还处于虚拟机外部存储空间里,要想执行它需要通过类加载器来加载到虚拟机的运行时内存空间里。关于类加载器不太想过多扩展,有兴趣珂查阅相关书籍资料。
  
  常见类加载器有:
  
  Bootstrap ClassLoader(启动类加载器:加载位于\lib 目录下的类文件,如rt.jar
  
  Extension ClassLoader(扩展类加载器): 加载位于\lib\ext目录下的类文件
  
  Application ClassLoader(应用程序类加载器):加载位于类路径(ClassPath)下的类文件
  
  总之,加载器的任务就是把字节码资源载入到虚拟机运行时环境里
  
  字节码验证 (Bytecode Verifier)
  
  当类加载器将新加载的字节码呈现给虚拟机时,首先由验证器来检查验证这些字节码。验证程序检查指令是否无法执行明显有害的操作。除系统类之外的所有类都需要经过验证。也可以使用命令-noverify选项来停用验证。
  
  字节码验证器主要验证如下几项:
  
  变量在使用前初始化
  
  不违反访问私有数据和方法的规则
  
  运行时堆栈不会溢出
  
  所有Java虚拟机指令的参数都是有效类型
  
  各种类型检
  
  总之,验证器的任务就是保证加载器载入的字节码资源的安全性,正确性
  
  3.2 解释器与JIT编译器
  
  解释器
  
  解释器(interpreter),是一种计算机程序,能够把高级编程语言一行一行解释 运行。
  
  划重点:一行一行运行,说白了就是效率低
  
  解释器每次运行程序时都要一行一行先转成另一种语言再作运行,因此解释器的程序运行速度比较缓慢。它不会一次把整段代码翻译出来,而是每翻译一行程序叙述就立刻运行,然后再翻译下一行,再运行,如此不停地进行下去。
  
  JIT编译器
  
  即时编译(Just-in-time compilation)是一种提高程序运行效率的方法。通常,程序在执行前全部被翻译为机器码。
  
  Java最初的版本没有JIT编译器,完全靠解释器来运行的,但是为了提升性能便引入了JIT编译器,
  
  重点说明:当我们说编译的时候基本上指的是上面的从源码到字节码的编译过程,而不是指JIT编译器
  
  JIT编译器工作阶段基本是java程序运行期的最后阶段了,它的工作是将加载的字节码转换为机器码。当使用JIT编译器时,硬件可以执行JIT编译器生成的机器码,而不是让JVM重复解释执行相同的字节码导致相对冗长的翻译过程。 这样可以带来执行速度的性能提升。
  
  什么时候触发即时编译?
  
  被多次调用的方法
  
  被多次执行的循环体
  
  上面两个条件又叫做热点代码,至于如何界定这个多次或者热点,Java提供了两种策略:
  
  热点探测: 虚拟机定期检查线程的栈顶,如果某个方法经常出现在栈顶 则推断为热点代码
  
  计数器: 统计方法的调用次数,维护一个计数器列表
  
  基于计数器来推断热点代码是HotSpot虚拟机采用的策略
  
  通常情况下,解释器和JIT编译器混合配合工作,而不是单独工作,这样可以做到互补提升整体性能。HotSpot 虚拟机的解释器JIT编译器架构如下图所示:
  
  HotSpot虚拟机中内置了两个即时编译器,分别称为Client Compiler和Server Compiler,或者简称为C1编译器和C2编译器,默认采用解释器与其中一个编译器直接配合的方式工作,程序使用哪个编译器,取决于虚拟机运行的模式,用户也可以使用“-client”或“-server”参数去强制指定虚拟机运行在Client模式或Server模式。
  
  4. 总结
  
  java 程序是如何运行的?
  
  首先需要把源代码(高级语言) 编译成虚拟机可执行的语言(字节码)
  
  其次,需要把字节码解释运行后者编译成操作系统级别的机器语言,用于执行函数调用(System call)
  
  Java是如何做到平台独立的?
  
  主要是因为字节码技术。我们可以把在Windows系统上编译生成的字节码文件放在Linux系统上去执行,反之亦可。
  
  虚拟机不在乎你是那个操作系统生成的字节码文件,他只在乎加载的这个.class字节码文件是否是正确的,安全的。
  
  虽然Java语言是平台独立的,但是虚拟机不行。每种操作系统都要下载对应的虚拟机,这主要是由于它最终调用的函数库以及线程模型不同。

© 著作权归作者所有

SEOwhywhy
粉丝 8
博文 155
码字总数 342404
作品 0
私信 提问
Linux ---> 监控JVM工具

JDK内置工具使用 jps(Java Virtual Machine Process Status Tool) 查看所有的jvm进程,包括进程ID,进程启动的路径等等。 jstack(Java Stack Trace) ① 观察jvm中当前所有线程的运行情况和线...

shking
2013/10/10
5.7K
0
如何在程序中控制java的单进程?

大家知道java程序运行时都是从main方法开始,产生一个java.exe的进程。但是这个java程序可以并发运行,这样在实际中就会有问题。那么程序本身能不能控制这种并发?也就是说我在第二次运行jav...

wangxin2008
2013/09/06
696
2
编写你的第一个HelloWorld

写在前面的话 因为Java基础是以后学习框架的基石,因此开个文集首先写写Java基础,本来想直奔基础知识的介绍,但是为了保证知识的完整性,因此从Java安装和运行“hello world”开始(虽然百度...

nanaFighting
2018/06/15
0
0
JVM基础:深入学习JVM堆与JVM栈

以前堆是干啥栈是干啥都知道,就是没连在一起想想。感觉讲的不错的一篇儿~~JVM栈解决程序的运行问题,即程序如何执行,或者说如何处理数据;JVM堆解决的是数据存储的问题,即数据怎么放、放在...

李星
2014/06/04
353
0
Java常识:IDE、JRE、JDK等字段的阐述

目前对这些最基本的常识都难以阐述,颇觉羞愧,果断整理出来。 IDE(Integrated Development 集成开发环境),集成开发环境(简称IDE)软件是用于程序开发环境的应用程序,一般包括代码编辑器、...

MiniBu
2014/05/06
883
0

没有更多内容

加载失败,请刷新页面

加载更多

查看线上日志常用命令

cat 命令(文本输出命令) 通常查找出错误日志 cat error.log | grep 'nick' , 这时候我们要输出当前这个日志的前后几行: 显示file文件里匹配nick那行以及上下5行 cat error.log | grep -C ...

xiaolyuh
27分钟前
5
0
六、Java设计模式之工厂方法

工厂方法定义: 定义一个创建对象的接口,但让实现这个接口的类来决定实例化哪个类,工厂方法让类的实例化推迟到子类中进行 类型:创建型 工厂方法-使用场景: 创建对象需要大量重复的代码 ...

东风破2019
33分钟前
5
0
win服务器管理遇到的一系列问题记录

有些小伙伴在使用iis7远程桌面管理工具的时候总是会遇到一系列的问题,下面就是为大家介绍一下服务器日常管理过程中出现的问题及我的解决办法和心得。希望能帮到大家。   拒绝服务器重新启...

1717197346
41分钟前
6
0
flutter 剪切板 复制粘贴

复制粘贴功能 import 'package:flutter/services.dart'; Clipboard.setData(ClipboardData(text:_text));Clipboard.getData;...

zdglf
43分钟前
4
0
如何保证消息的可靠性传输?或者说,如何处理消息丢失的问题?

面试题 如何保证消息的可靠性传输?或者说,如何处理消息丢失的问题? 面试官心理分析 这个是肯定的,用 MQ 有个基本原则,就是数据不能多一条,也不能少一条,不能多,就是前面说的重复消费...

米兜
44分钟前
3
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部