Java的加载与执行
Java程序的运行包括两个非常重要的阶段:
一.编译阶段注意:Java程序员直接编写的Java代码(普通文本)是无法被JVM直接识别的,必须通过编译将“普通文本代码”变为“字节码“
第一步:程序员需要在计算机硬盘中任意位置创建一个.java扩展名的文件,该文件被称为 java源文件,源文件当中编写的是java源代码/源程序。java源代码必须符合java的语法规则。第二步:java程序员需要使用JDK当中自带的javac.exe命令进行java程序的编译。
Javac使用规则?javac java源文件的路径javac在哪用?在DOS命令窗口中使用,win+r打开“运行窗口”,输入cmd–注:一个java源文件可以编译生成多个.class文件编译阶段主要检查java源代码是否符合Java语法。符合java语法则能生成以.class结尾的字节码文件,不符合java语法则无法形成字节码文件。字节码文件不是纯粹的二进制,无法在操作系统中直接执行。编译结束之后,可以将class文件拷贝到其他操作系统当中(java跨平台性)。字节码文件/class文件是最终要执行的文件,所以说class文件生成之后,java源文件/.java文件删除并不会影响程序执行。但是一般不删除,最后可能执行效果不是想要的,只能将java源文件重新修改,然后java源文件重新生成class文件,运行class文件生成新的效果。
二.运行阶段JDK安装之后,除了自带javac.exe之外,还有另一个工具/命令,叫java.exe,java.exe主要负责运行阶段。
java.exe在哪里用?在DOS窗口中使用,win+r打开“运行窗口”,输入cmd
java.exe怎么用?java 类名 注意与javac命令的区分例如:硬盘上有一个A.class文件,则:java A硬盘上有一个b.class文件,则:java b注意不能写成:java A.class(错误方式)运行阶段过程步骤:1.打开DOS命令窗口2.输入:java A3.java.exe命令会启动Java虚拟机(JVM),JVM会启动类加载器ClassLoader。4.ClassLoader会去硬盘上搜索A.Class文件,找到该文件则将该文件字节码文件装载到JVM当中。5.JVM将A.Class字节码文件解释成二进制10101010这样的数据。6.然后操作系统执行二进制和底层硬件平台进行交互。
编译阶段和运行阶段可以在不同的操作系统上完成,因为Java具有跨平台性
Java源代码编译后可以删除源代码,只留下字节码,因为源代码不参与程序的执行过程参与执行过程的是字节码,但是最好不要删除源代码
放源代码的文件扩展名必须是xxx.java编译生成的字节码文件扩展名是xxx.class
注意:一个Java源文件是可以编译生成多个class文件的,最终运行的是class文件字节码文件不是二进制文件,如果是二进制,就不需要JVM,因为操作系统可以直接执行二进制文件
Java程序从开发到最终运行经历的过程:编译器:(可以在Windows上)第一步:在硬盘中新建xxx.java文件第二步:使用文本编辑器打开xxx.java文件第三步:在xxx.java文件中编写“符合Java语法规则”的源代码第四步:保存第五步:使用编译器(javac【JDK安装后自带】)对xxx.java文件进行编译第六步:如果xxx.java文件是符合语法规则的,编译会通过;如果xxx.java文件中编写的源代码违背了语法规则,那么编译器会报错,编译器报错后class文件不会生成,只有编译通过了才会生成class字节码文件,并且一个Java源文件是可以生成多个class文件的。(编译实质上是检查语法)
运行期:(可以在widows上,也可以在其他的OS上)第七步:如果是在Linux上运行需要将Windows上生成的class文件拷贝过去,不需要拷贝源代码,真正在Linux运行的是字节码。(但是不要删除源代码)第八步:使用JDK自带的一个命令/工具:Java(负责运行的命令/工具)执行字节码第九步:JVM会将字节码文件装载进去,然后JVM对字节码进行解释(解释器会将字节码解释为二进制)第十步:JVM会将生成的二进制码交给OS,操作系统就会执行二进制码和硬件进行交互
注:在以上过程中,有两个重要命令:javac命令,负责编译java命令,负责运行xxx.java源文件经过编译之后生成了A.class、B.class、C.class等文件其中称A是一个类,B是一个类,C是一个类;A,B,C是类的名字
以上过程中,程序员的工作:新建Java文件打开Java文件写Java源代码保存javac命令编译java命令运行
即:编写-编译-运行
一行JAVA代码如何运行起来?
在程序员的世界中,你总会听到一句“PHP是世界上最好的语言”的调侃。然而在你进入软件程序开发之后,你会发现即使开发语言千千万,最盛行的还是JAVA。从淘宝的技术变迁中我们可以见一些端倪,早期电商刚起来的时候,那会儿的互联网还很简单,使用PHP+Mysql+Apache+Linux就可以快速搭建起一套电商系统,但随着电商平台、支付平台的完善,网上购物开始变得简单,越来越多的人使用淘宝购物了,淘宝的技术架构也开始不断的升级,增加服务器数量来提高系统可用性。
通过运维手段扩充资源是一种方式,治标不治本,最根本的原因还是在于PHP这种语言可扩展性不够,用户量十万、百万、千万的时候都还能支撑,但到了上亿、亿万的时候怎么扩展都不行了。于是淘宝系统开始一点点的前后端分离,后端使用JAVA语言开发,逐渐迁移业务。现在我们所使用的淘宝系统,80%以上的后端程序都是Java开发,可见笑到最后才是赢家啊。不过JAVA语言的上手难度就比PHP、前端高很多了,所以今天我们给大家讲解下一行JAVA代码到底是如何运行起来的,JAVA后浪们可以以此为入门Java的基础,开启Java开发、人生赢家之路。
Java是一种半解释型语言,相对的有解释型语言Python&PHP、编译型语言C&C++。解释型语言说的是只需要在客户端输入代码后就可以运行起来,实时看到结果,编译型语言说的是源代码需要进行构建编译成二进制文件才能在机器运行起来,半解释型语言介于其中,它把输入的代码进行编译,编译后在JVM虚拟机中运行(注:JVM虚拟机是在实际的机器中运行的)。半解释型语言的好处就是可以跨平台,一次编译,多次执行。
我们通过下面这边Java程序,来讲明Java程序从编译到最后运行到整个流程。JVM运行Java程序有两种方式,分别是jar包和Class类文件,jar包是偏上层的方式,把所有程序都打包成一个jar包,便于交付测试人员测试、运维人员发布,它的运行逻辑是通过java.exe找到java自带的GetMainClassName函数,该函数获取JNIENV实例,并调用JarFileJNIENV实例中的GetMainfest()函数获取MainClass函数,Main函数再调用Java.c中的LoadClass方法加载主类。
而Class方式则是越过上层,直接通过main函数调用Java.c中的LoadClass方法装载类。所以说jar运行的方式本质上也是class类运行的方式,因此我们来关注如何类方式如何加载运行就好了。下面代码想实现的功能是打印Code这个字符,整体代码如下。我们先定义了一个类HelloJava,在这个类新建了一个对象去打印Code字符,而这个对象又调用了类Product.java
在整个代码的运行中,包含两步,第一步是编译,第二步是运行。源文件创建完之后,使用javac就可以编译.java程序,程序会被编译成.class文件,使用java命令就可以运行.class文件。编译后的文件有代码中出现过的类名&变量名&方法引用名、类中各个方法的字节码,它们分别存储在常量池、方法字节码中。
在Java程序的编译过程中,如果该类所依赖的类还没有被编译,编译器就会先编译被依赖的类,如果依赖类编译了则直接引用。在Java类的运行中,包含加载和运行两个步骤。.class文件就是通过类加载器到jvm当中的。在Java中默认有三种类加载器,从下往上依次是自定义类加载器UserClassLoader(负责加载自定义的class文件)、应用类加载器AppClassLoader(负责加载classpath指定的jar包和目录中的class文件)、扩展类加载器ExClassLoader(负责加载Java平台中扩展功能的jar包)、启动类加载器BootstrapClassLoader(负责加载$JAVA_Home中jre/lib/rt.jar中所有的class类)。当AppClassLoader接收到一个类加载命令后,它不会自己先去加载,而是给到扩展类加载器,同样扩展类加载器自己也不会先去加载类,而是把它给到启动类加载器去加载,如果失败再层层往下传递。所以Java是动态在加载类。
回到我们刚刚的例子中,在编译好Java程序后,我们得到HelloJava.class文件,在终端我们输入javaHelloJava,系统就会启动一个JVM进程,JVM进程从classpath的路径中寻找命名为HelloJava.class的二进制文件,将HelloJava的类加载信息加载到运行时数据区的方法区,找到HelloJava的主函数入口,执行Main函数。Main函数的第一条命令是Productproduct = newProduct(“Code”),它需要JVM创建一个Product对象,但此时方法区中没有没有Product类的信息,于是JVM加载Product类,把Product类的类型信息放在方法区中。加载完了Product类之后,JVM虚拟机在堆区为新的Product实例分配内存,初始化类。在调用product.printName()方法的时候,JVM根据Product引用找到Product对象,根据Product对象持有的引动定位到方法区中的Animal类的类型信息方法表,获取printName()函数的字节码地址,运行printName()函数,打印出来“Code”。
微观的编译执行介绍完了,我们来看看中观的执行。在介绍Java是解释型语言时,我们有讲到JVM是跨平台执行的,也就是一份Java代码编译之后,可以在Linux、unix、Windows、Macos等操作系统平台中执行。我们一起来看看是如何实现的呢?在Java程序运行中有三个概念,JVM、JDK、JRE。
- 所谓JVM就是Javavirtual Machine,Java虚拟机,执行Java代码;
- 所谓JDK是指的JavaDevelopment kit,Java开发工具包,Java开发人员使用;
- 所谓JRE就是JavaRuntimeEnvironment,Java运行时环境。
JVM属于JRE,JRE属于JDK。在JDK的安装中,有不同的版本,比如Linuxx86、Windowsx64,只要安装了JDK之后,就由JDK来区分操作系统,JVM是运行在操作系统之上,区分操作系统的任务就是由JDK来完成的,只要你的电脑装了JDK,任何一份Class字节码都会运行在JVM中,JVM又可以运行在任意操作系统中,从而实现了“跨平台一次编译,多次执行”。
讲完了中观的执行,我们来看看宏观执行。我们程序员在写Java代码时,都会把程序编译成jar包,通过jar包来运行程序。一个jar包代表了一个功能模块的实现,如果某个jar包有我们想要使用的功能,就在程序中引用就好。然而业务功能在开发实现时可运行依赖的jar包很多,如果把每个功能所实现的jar包都放在自己的jar包中,就会非常的浪费资源和运行效率。这时候我们可以把程序依赖的jar包都放在一个单独的文件夹中,然后修改jar包中“META-INF”目录下的“MANIFEST.MF”清单文件即可。在manifest文件中,我们指定Manifest文件的版本,运行主类的名称,程序所依赖的jar包的Classpath路径都写明清楚,Java程序执行时加载manifest文件即可。
本文详细的介绍了一行JAVA代码是如何在JVM系统中运行起来的,对于有志加入互联网行业,使用Java语言开发贡献力量的朋友们来说,可以在初学时深刻的理解体会到Java代码时怎么运行起来的、JDK&JRE&JVM是什么?在面试的时候也能比较轻松从容的回到面试官问题,在带新人的时候也可以装一把大佬。码字不易,赶紧收藏起来这份Java宝典吧,如果你愿意,点个在看和喜欢,把它也传递给你的伙伴们喔~
本文作者及来源:Renderbus瑞云渲染农场https://www.renderbus.com
文章为作者独立观点不代本网立场,未经允许不得转载。