java 程序执行原理

转自:http://blog.csdn.net/walkingmanc/article/details/6369487
java 应用可以打包成jar 格式, jar格式其实只是一种很普通的压缩格式,与zip格式一样,只不过是它会在压缩文件的目录结构中增加一个META-INF/ MANIFEST.MF 的元文件。

我们知道,经过编译的字节码class文件可以直接放到java虚拟机去解释执行(JIT方式), 我们通过在命令行调用“java class文件的路径”就可以使用jvm(java.exe/javaw.exe)来解释执行这些字节码文件。

我们知道,java源代码(.java文件)经过java编译器javac编译以后,会得到java的字节码的中间语言文件,也就是我们通常所说的类文件(.class文件),这些类文件会按照java源文件的包结构分目录存放,jar 命令的作用就是按照这种包目录结构打包这些字节码的class文件,形成一个jar包,并且增加一个META-INF/ MANIFEST.MF 的元文件 。这样打包的jar文件的确是包含了按照包目录结构存放的字节码class文件,但是这时候如果你在命令行:jar -jar a.jar 的话,会提示你指定一个主类,这是因为,虽然jar包里面包含了按照包目录结构存放的字节码class文件,但是却并不知道主类(含有 public static void main(String[])入口 方法的类)的位置,所以需要你手动的指定主类,然后才可以开始执行。

当然, 只需要在打包jar文件的时候,将主类的信息包含进去了以后,再:jar -jar a.jar 的话, 就不需要手动的指出主类是哪个类了。进行如下操作:

新建一个.mf文件,名字任意,例如:manifest.mf ,在里面指定主类是哪个类,即:写入一行 :Main-Class: test.Test。然后,打包: jar cvfm test.jar manifest.mf test; 这样打成的包test.jar里面就已经包含主类是哪个类的信息了。这样的话,在命令行里面直接执行:jar -jar test.jar 就可以运行该应用了,这种情况下, jvm会去这个class文件的包中寻找入口函数如何进入执行。

我们发布java应用的时候,直接发布按照包结构存放的class文件夹显然是很不方便的,一般需要打包成jar文件来发布。 就好象我们将java应用部署到tomcat web 服务器上去的时候一样, 虽然直接将按照包结构存放的class文件夹放过去是可以的,但是通常还是要打包成jar包的格式来部署,因为这样文件的数量会少,便于管理部署的。

将命令行的“java -jar jar文件的路径” 写到一个.bat文件里面 或者linux下面的shell文件里面,这样就可以通过执行.bat文件或者shell脚步来执行java 应用程序了,也就是说可以直接双击windows下的.bat文件或者linux下的.sh文件来运行java应用程序了。

一般在安装jre (Java Runtime Environment) 的时候,安装文件会将 .jar 文件映射给 javaw.exe 打开,如果没有关联,也可以手动的通过文件夹选项来手动关联,关联以后,当用户在windows下双击jar文件的时候,资源管理器就会调用javaw来运行该jar文件,这样就可以实现双击运行jar文件了。

jdk 相关过程原理分析。

我们知道,在jdk的bin目录下有很多的exe文件,例如java.exe, javac.exe, javadoc.exe等。 这些exe文件格式实质上是windows操作系统下的可执行文件格式(在dos下还有一种可执行可是是.com后缀的格式,不过现在已经不常见了),它们是由C语言写成的.c文件经过编译后生成的。例如:java.exe对应的源码就是java.c文件。java.c的main入口函数中会调用函数:CreateExecutionEnvironment,该函数中会查找jre路径,然后根据jvm.cfg配置文件配置的虚拟机动态链接库(jvm.dll)路径参数装载jvm.dll动态连接库,也就是加载java虚拟机(java虚拟机是C++写的,也有部分C代码),然后初始化jvm.dll(所有的dll都是本地语言写成的), 并挂接到JNIEnv(JNI调用接口)实例,最后调用JNIEnv实例装载并处理class类。
由上面的分析我们可以看到,windows操作系统下的exe文件大部分情况下是使用windows本地语言所写的代码编译而成的的,这些exe文件用于完成一定的功能,例如java.exe, 可以用来查找并加载jvm.dll ,然后通过调用jvm.dll 的 接口来加载java的字节码中间语言文件.class文件,并启动java应用程序。 或者完成其它的一些功能等。exe文件也可能是本地语言代码生成的exe文件与jar包压缩而成的。

更加方便的方法是,将jar做成exe。例如eclipse 就是一个java 应用程序,就采用了 使用exe来wrapper。

wrapper 基本原理: 在本地化语言(C或者C++等)代码中调用jvm.dll,然后通过jvm.dll提供的接口加载压缩在一起的jar包中的主类class的入口方法( static void main(String args[]), 从而启动java应用程序,这种加壳方式形成的java应用的exe文件在启动的时候会表现为一个exe进程,这种方式更常见。(形式是一个由本地化语言exe和jar包一起压缩而成的一个exe文件);

也可以在本地化语言(C或者C++等)代码中调用java.exe/javaw.exe进程(java.exe进程会执行前一种方法的步骤来完成jvm.dll的加载)来加载jvm.dll,然后通过jvm.dll提供的接口加载压缩在一起的jar包中的主类class的入口方法( static void main(String args[]), 从而启动java应用程序,这种加壳方式形成的java应用的exe文件在启动的时候会表现为一个exe进程和一个javaw进程。(形式是一个由本地化语言exe和jar包一起压缩而成的一个exe文件,当然,也可以选择不将jar文件和本地exe文件压缩在一起);

这两种根本上都是通过本地代码来加载java虚拟机,然后在本地代码中通过调用jvm.dll的接口来完成class主文件的加载和java应用的启动的。

具体来说,也有很多种方法。

可以使用c语言等本地语言创建一个本地代码(编译后生成exe文件),在该本地代码中加载jvm的动态库,并通过动态库的接口来在本进程内启动java虚拟机,即调用JNI_CreateJavaVM这个导出函数来创建Java虚拟机,得到JNIEnv指针,然后调用FindClass查找Main Class,之后调用GetStaticMethodID方法得到main方法,并执行main方法。 编译该本地代码得到一个exe文件。

将Java应用程序的class目录结构打包为jar文件,并与本地代码exe文件合并:在Dos提示符下执行copy命令:

C:/>copy test.exe+test.jar test.exe
其实,就是将Java打包文件追加到exe文件尾部。打开文件属性对话框,可看到有“压缩文件”属性页。
老牌的JBuilder.exe开发工具编译生成的exe文件即采用如下方式生成。
后记:大家在使用Eclipse 3.2和Eclipse 3.3时,在任务管理器中会看到二者的不同。
Eclipse 3.2是先启动Eclipse.exe文件,然后由Eclipse.exe启动Javaw.exe文件来创建虚拟机。
Eclipse 3.2在任务管理器中显示为Eclipse.exe和javaw.exe两个进程。
Eclipse 3.3在任务管理器中显示为Eclipse.exe一个进程。
从上面可以看出,Eclipse 3.2和Eclipse 3.3采用了不同的虚拟机加载方式。

Eclipse 3.2采用创建子进程的方法调用javaw.exe来启动,在windows下面可以用CreateProcess方法,此种方法较简单,具体可参见Eclipse源码。

Eclipse 3.3加载java虚拟机的另外一种方法是加载jvm的动态库,并通过动态库的接口来在本进程内启动java虚拟机。本文开头即采用的第二种方法。

写一个exe,调用java.exe来启动一个进程。
这很简单,启动vc,建一个win32 project,WinMain里使用ShellExecute函数即可,主要代码是:

#include "stdafx.h"
#include "resource.h"

int APIENTRY WinMain(HINSTANCE hInstance,
HINSTANCE hPrevInstance,
LPSTR lpCmdLine,
int nCmdShow)
{
ShellExecute(NULL,"open",".//jre6//bin//javaw.exe","-ea -Dfile.encoding=GB18030 -Xmx600M -splash:res//splash.png -classpath /"./lib/*/" com.zms.laser.uis.Starter",".//",SW_SHOWNORMAL);
return 0;
}
要给编译出来的可执行程序一个图标,很简单,只要添加一个icon的资源,让这个icon的id最小即可。编译器会把具有最小id的icon当作最终exe的图标。

所有的给java 应用 加 exe 外壳的 工具软件 采用的都是以上原理中的一种。

时间: 2024-10-23 04:26:01

java 程序执行原理的相关文章

java程序运行原理

一.JRE.JDK.JVM 要了解java程序运行原理,首先需要了解知道jre.jdk.jvm这三者是什么,他们之间又有什么联系. JRE(JavaRuntimeEnvironment,Java运行环境),也就是Java平台.所有的Java 程序都要在JRE下才能运行. JDK(Java Development Kit,java开发工具包)是程序开发者用来来编译.调试java程序用的开发工具包.JDK的工具也是Java程序,也需要JRE才能运行.为了保持JDK的独立性和完整性,在JDK的安装过程

java 程序执行输出有两种简单方式

java 程序执行输出有两种简单方式: 1. System.out.println("需要输出的内容"): 该方法可参看运行一个简单的Java程序 结果图: 2. System.out.print("需要输出的内容"): 1 public class HelloWorld 2 { 3 //Java程序的入口方法,程序将从这里开始运行 4 public static void main(String[] args) 5 { 6 //向控制台打印一条语句 7 Syste

java程序执行SQL脚本文件

首先引入ibatis-common-2.jar包 import com.ibatis.common.jdbc.ScriptRunner; import com.ibatis.common.resources.Resources; jpetstore测试代码如下: package com.ibatis.jpetstore.test; import java.sql.DriverManager; import java.util.Properties; import com.ibatis.commo

java程序执行sql文件

List<String> sqlFileList = new ArrayList<String>();从文件读放内容到按分号放到sqlFileListpublic List<String> readSqlFiles(List<String> fileNameList) {List<String> sqlList = new ArrayList<String>();for (String fileName : fileNameList)

java程序执行过程&amp;基本数据类型

1. 程序load到内存. 2. 找到程序入口方法(main())开始执行. 3. 程序在内存中的存放 3.1 代码段(code segment)--------存放代码 3.2 数据段(data segment)--------存放静态变量,字符串常量 3.3 栈(stack)                ----------存放局部变量 3.4 堆(heap)                -----------存放new出来的东西 ----------------------------

深入理解Java程序执行顺序

下面将从一道阿里巴巴试题详细分析Java程序执行顺序. 阿里巴巴试题 public class Test { public static int k = 0; public static Test t1 = new Test("t1"); public static Test t2 = new Test("t2"); public static int i = print("i"); public static int n = 99; publi

Java程序运行原理分析

class文件内容 class文件包含Java程序执行的字节码 数据严格按照格式紧凑排列在class文件的二进制流,中间无分割符 文件开头有一个0xcafebabe(16进制)特殊的标志 JVM运行时数据区 线程独占: 每个线程都会有它独立的空间,随线程的生命周而创建和销毁 线程共享: 所有线程都能访问这块内存数据,随虚拟机或GC而创建和销毁 方法区 方法区是各个线程共享的内存区域 用于存储已被虚拟机加载的类信息, 常量,静态变量, 即时编译后的代码等数据 虽然Java虚拟机规范把方法区描述为堆

python基础-程序执行原理

1.程序执行原理: 2.python的执行原理: 3.程序的作用:处理数据 4.变量就是用来存储数据的 原文地址:https://www.cnblogs.com/leading-net/p/12550039.html

Java程序执行Linux命令(JSP运行其他程序)

java程序中要执行linux命令主要依赖2个类:Process和Runtime 首先看一下Process类: ProcessBuilder.start() 和 Runtime.exec 方法创建一个本机进程,并返回 Process 子类的一个实例,该实例可用来控制进程并获得相关信息.Process 类提供了执行从进程输入.执行输出到进程.等待进程完成. 检查进程的退出状态以及销毁(杀掉)进程的方法. 创建进程的方法可能无法针对某些本机平台上的特定进程很好地工作,比如,本机窗口进程,守护进程,M