iis服务器助手广告广告
返回顶部
首页 > 资讯 > 后端开发 > Python >Java虚拟机调用Java主类的main()方法
  • 849
分享到

Java虚拟机调用Java主类的main()方法

2024-04-02 19:04:59 849人浏览 安东尼

Python 官方文档:入门教程 => 点击学习

摘要

目录鸠摩 在前一篇 第1篇关于Java虚拟机HotSpot,开篇说的简单些 中介绍了call_static() 、call_virtual()等函数的作用,这些函数会调用JavaCa

鸠摩

在前一篇 第1篇关于Java虚拟机HotSpot,开篇说的简单些 中介绍了call_static()call_virtual()等函数的作用,这些函数会调用JavaCalls::call()函数。我们看Java类中main()方法的调用,

调用栈如下:


JavaCalls::call_helper() at javaCalls.cpp 
os::os_exception_wrapper() at os_linux.cpp 
JavaCalls::call() at javaCalls.cpp
jni_invoke_static() at jni.cpp 
jni_CallStaticVoidMethod() at jni.cpp 
JavaMain() at java.c
start_thread() at pthread_create.c
clone() at clone.S

这是Linux上的调用栈,通过JavaCalls::call_helper()函数来执行main()方法。栈的起始函数为clone() ,这个函数会为每个进程(Linux进程对应着Java线程)创建单独的栈空间,这个栈空间如下图所示。

在Linux操作系统上,栈的地址向低地址延伸,所以未使用的栈空间在已使用的栈空间之下。图中的每个蓝色小格表示对应方法的栈帧,而栈就是由一个一个的栈帧组成。native方法的栈帧、Java解释栈帧和Java编译栈帧都会在***域中分配,所以说他们寄生在宿主栈中,这些不同的栈帧都紧密的挨在一起,所以并不会产生什么空间碎片这类的问题,而且这样的布局非常有利于进行栈的遍历。上面给出的调用栈就是通过遍历一个一个栈帧得到的,遍历过程也是栈展开的过程。后续对于异常的处理、运行jstack打印线程堆栈、GC查找根引用等都会对栈进行展开操作,所以栈展开是后面必须要介绍的。

下面我们继续看JavaCalls::call_helper()函数,这个函数中有个非常重要的调用,如下:


// do call
{
    JavaCallWrapper link(method, receiver, result, CHECK);
    {
      HandleMark hm(thread);  // HandleMark used by HandleMarkCleaner
      StubRoutines::call_stub()(
         (address)&link,
         result_val_address,              
         result_type,
         method(),
         entry_point,
         args->parameters(),
         args->size_of_parameters(),
         CHECK
      );
 
      result = link.result();  
      // Preserve oop return value across possible gc points
      if (oop_result_flag) {
        thread->set_vm_result((oop) result->get_jobject());
      }
    }
}

调用StubRoutines::call_stub()函数返回一个函数指针,然后通过函数指针来调用函数指针指向的函数。通过函数指针调用和通过函数名调用的方式一样,这里我们需要清楚的是,调用的目标函数仍然是C/C++函数,所以由C/c++函数调用另外一个C/C++函数时,要遵守调用约定。这个调用约定会规定怎么给被调用函数(Callee)传递参数,以及被调用函数的返回值将存储在什么地方。

下面我们就来简单说说Linux X86架构下的C/C++函数调用约定,在这个约定下,以下寄存器用于传递参数:

  • 第1个参数:rdi c_rarg0
  • 第2个参数:rsi c_rarg1
  • 第3个参数:rdx c_rarg2
  • 第4个参数:rcx c_rarg3
  • 第5个参数:r8 c_rarg4
  • 第6个参数:r9 c_rarg5

在函数调用时,6个及小于6个用如下寄存器来传递,在HotSpot中通过更易理解的别名c_rarg*来使用对应的寄存器。如果参数超过六个,那么程序将会用调用栈来传递那些额外的参数。

数一下我们通过函数指针调用时传递了几个参数?8个,那么后面的2个就需要通过调用函数(Caller)的栈来传递,这两个参数就是args->size_of_parameters()CHECK(这是个宏,扩展后就是传递线程对象)。

所以我们的调用栈在调用函数指针指向的函数时,变为了如下状态:

右边是具体的call_helper()栈帧中的内容,我们把threadparameter size压入了调用栈中,其实在调目标函数的过程还会开辟新的栈帧并在parameter size后压入返回地址和调用栈的栈底,下一篇我们再详细介绍。先来介绍下JavaCalls::call_helper()函数的实现,我们分3部分依次介绍。

1、检查目标方法是否"首次执行前就必须被编译”,是的话调用JIT编译器去编译目标方法;

代码实现如下:


void JavaCalls::call_helper(
 JavaValue* result, 
 methodHandle* m, 
 JavaCallArguments* args, 
 TRAPS
) {
  methodHandle method = *m;
  JavaThread* thread = (JavaThread*)THREAD;
  ...
 
  assert(!thread->is_Compiler_thread(), "cannot compile from the compiler");
  if (CompilationPolicy::must_be_compiled(method)) {
    CompileBroker::compile_method(method, InvocationEntryBci,
                                  CompilationPolicy::policy()->initial_compile_level(),
                                  methodHandle(), 0, "must_be_compiled", CHECK);
  }
  ...
}

对于main()方法来说,如果配置了-Xint选项,则是以解释模式执行的,所以并不会走上面的compile_method()函数的逻辑。后续我们要研究编译执行时,可以强制要求进行编译执行,然后查看执行过程。

2、获取目标方法的解释模式入口from_interpreted_entry,也就是entry_point的值。获取的entry_point就是为Java方法调用准备栈桢,并把代码调用指针指向method的第一个字节码的内存地址。entry_point相当于是method的封装,不同的method类型有不同的entry_point

接着看call_helper()函数的代码实现,如下:


address entry_point = method->from_interpreted_entry();

调用methodfrom_interpreted_entry()函数获取Method实例中_from_interpreted_entry属性的值,这个值到底在哪里设置的呢?我们后面会详细介绍。

3、调用call_stub()函数,需要传递8个参数。这个代码在前面给出过,这里不再给出。下面我们详细介绍一下这几个参数,如下:

  • (1)link 此变量的类型为JavaCallWrapper,这个变量对于栈展开过程非常重要,后面会详细介绍;
  • (2)result_val_address 函数返回值地址;
  • (3)result_type 函数返回类型;
  • (4)method() 当前要执行的方法。通过此参数可以获取到Java方法所有的元数据信息,包括最重要的字节码信息,这样就可以根据字节码信息解释执行这个方法了;
  • (5)entry_point HotSpot每次在调用Java函数时,必然会调用CallStub函数指针,这个函数指针的值取自_call_stub_entry,HotSpot通过_call_stub_entry指向被调用函数地址。在调用函数之前,必须要先经过entry_point,HotSpot实际是通过entry_point从method()对象上拿到Java方法对应的第1个字节码命令,这也是整个函数的调用入口;
  • (6)args->parameters() 描述Java函数的入参信息;
  • (7)args->size_of_parameters() 参数需要占用的,以字为单位的内存大小
  • (8)CHECK 当前线程对象。

到此这篇关于Java虚拟机调用Java主类的main()方法的文章就介绍到这了,更多相关Java虚拟机调用main()方法内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

--结束END--

本文标题: Java虚拟机调用Java主类的main()方法

本文链接: https://www.lsjlt.com/news/156331.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • Java虚拟机调用Java主类的main()方法
    目录鸠摩 在前一篇 第1篇关于Java虚拟机HotSpot,开篇说的简单些 中介绍了call_static() 、call_virtual()等函数的作用,这些函数会调用JavaCa...
    99+
    2024-04-02
  • java的main方法中调用spring的service方式
    目录main方法调用spring的servicemain方法调用spring的dao service方法main方法调用spring的service 将业务层类配置到Spring中:...
    99+
    2024-04-02
  • 云虚拟主机部署java的方法是什么
    云虚拟主机部署Java的方法有以下几种:1. 安装Java Runtime Environment (JRE):首先需要在虚拟主机上...
    99+
    2023-09-12
    云虚拟主机 java
  • java虚拟主机怎么使用
    使用Java虚拟主机(JVM)可以让您在共享服务器上运行Java应用程序。以下是使用Java虚拟主机的步骤:1. 选择一个可靠的虚拟...
    99+
    2023-05-17
    java虚拟主机 虚拟主机
  • java虚拟主机安装部署的方法是什么
    1. 安装Java虚拟机:首先需要安装Java虚拟机(JVM),可以从官方网站下载适合操作系统的JVM安装包。2. 下载Tomcat...
    99+
    2023-06-12
    java虚拟主机 虚拟主机
  • java的main方法中如何调用spring的service方式
    小编给大家分享一下java的main方法中如何调用spring的service方式,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!main方法调用spring的se...
    99+
    2023-06-21
  • Java虚拟机之类加载
    目录一、类加载流程1.1 类加载条件1.2 加载1.3 验证1.4 准备1.5 解析1.6 初始化二、ClassLoader2.1 Class...
    99+
    2024-04-02
  • Java虚拟机内存优化的方法
    这篇文章主要介绍“Java虚拟机内存优化的方法”,在日常操作中,相信很多人在Java虚拟机内存优化的方法问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Java虚拟机内存优化的方法”的疑惑有所帮助!接下来,请跟...
    99+
    2023-06-17
  • Java JVM虚拟机调优详解
    目录jmap查看内存信息jstackjinfo查看jvm系统参数Jstat查看堆内存使用和类加载的数量信息内存泄漏jmap查看内存信息 jmap histo /pid > ./...
    99+
    2024-04-02
  • Java虚拟机JVM类加载机制(从类文件到虚拟机)
    目录一、类加载机制简介二、类加载机制过程 2.1、加载(Load)2.2、连接(Linking)2.3、初始化(Initialize)三、类加载器Classloader&n...
    99+
    2024-04-02
  • 虚拟主机如何部署java
    要部署Java应用程序,可以按照以下步骤在虚拟主机上进行操作:1. 首先,确保您的虚拟主机支持Java应用程序的部署。通常,虚拟主机...
    99+
    2023-09-12
    虚拟主机 java
  • java虚拟主机为什么贵
    java虚拟主机贵的原因有:java虚拟主机贵,是由于java虚拟主机环境需要单独配置,服务器硬件、软件成本比普通高,维护成本也比其他虚拟主机要高,等多方面因素导致java虚拟主机贵。具体分析如下:市场小java虚拟主机,主要满足java语...
    99+
    2024-04-02
  • 虚拟主机的备案方法
    这篇文章将为大家详细讲解有关景安虚拟主机的备案方法,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。虚拟主机如何备案?虚拟主机备案,相当于网站在工信部系统中进行登记,相当于给网站做实名认证。备案的目的是为了防...
    99+
    2023-06-06
  • 如何理解JAVA虚拟主机
    今天就跟大家聊聊有关如何理解JAVA虚拟主机,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。其实虚拟主机也可以称之为网站空间,就是把一台运行在互联网上的物理服务器,划分为多个虚拟服务器...
    99+
    2023-06-07
  • 虚拟主机怎么运行java
    要在虚拟主机上运行Java,您需要确保虚拟主机上安装了Java运行时环境(JRE)或Java开发工具包(JDK)。然后按照以下步骤进...
    99+
    2023-08-31
    虚拟主机 java
  • Java中的main()方法怎么用
    这篇文章将为大家详细讲解有关Java中的main()方法怎么用,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。Java中的main()方法详解在Java中,main()方法是Java应用程序的入口方法,也就...
    99+
    2023-06-03
  • java虚拟机之JVM调优详解
    目录JVM常用命令行参数1. 查看参数列表2. 基本参数说明:3. 扩展参数说明:虚拟机参数分类什么是调优1.调优步骤:2.调优案例2.1案例一2.2案例二JVM优化总结JVM常用命...
    99+
    2024-04-02
  • java虚拟机参数如何调整
    Java虚拟机(JVM)参数的调整可以优化应用程序的性能和稳定性。以下是一些常见的Java虚拟机参数以及如何进行调整: 堆内存大...
    99+
    2024-04-09
    java
  • Java虚拟机安装的方法是什么
    安装Java虚拟机(JVM)的方法如下:1. 下载Java Development Kit(JDK):首先,你需要下载适用于你操作系...
    99+
    2023-09-14
    Java
  • 虚拟云主机怎么部署java
    要在虚拟云主机上部署Java应用程序,可以按照以下步骤进行操作:1. 登录到云主机的控制台或使用SSH工具连接到云主机。2. 确保云...
    99+
    2023-09-16
    虚拟云主机 java
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作