C++的extern关键字在HotSpot VM中的重要应用

c++,extern,关键字,hotspot,vm,重要,应用 · 浏览次数 : 7

小编点评

**1. extern在C/C++语言中表示函数和全局变量作用范围(可见性)的关键字** `extern` 在C/C++语言中用于声明函数和全局变量,其作用范围(可见性)在本模块或其它模块中可用。 **2. C++中引用C语言中的函数和变量** 在包含C语言头文件时,需要使用 `extern \"C\"` 来处理。例如: ```c #include <jni.h>#ifndef _Included_com_mprofiler_Test#define _Included_com_mprofiler_Test#ifdef __cplusplusextern \"C\" {#endifJNIEXPORT void JNICALL Java_com_mprofiler_Test_helloWorld (JNIEnv *, jobject, jstring) ``` `__cplusplus`宏用于支持用C或C++写native方法。 **3. HotSpot VM中的兼容性处理** HotSpot VM支持两种在C或C++中使用native方法的兼容性处理: - **`extern `在jni.h头文件中**:`__cplusplus`判断 用于避免在C++中使用C语言。 - **`extern \"C\"在mutexLocker.hpp文件中**:`Mutex`的外部变量 `SystemDictionary_lock` 在 `mutexLocker.cpp` 中定义,在编译和连接时系统会了解该变量已在别处定义。 **4. jni.h头文件的用法** jni.h头文件包含了对C语言中各种函数和变量的定义,允许C++程序员在使用C语言时访问HotSpot VM内部的功能。例如: ```c #include <jni.h>#ifndef _Included_com_mprofiler_Test#define _Included_com_mprofiler_Test#ifdef __cplusplusextern \"C\" {#endifJNIEXPORT void JNICALL Java_com_mprofiler_Test_helloWorld (JNIEnv *, jobject, jstring) ```

正文

extern关键字有两个用处:

(1)extern在C/C++语言中表示函数和全局变量作用范围(可见性)的关键字,这个关键字会告诉编译器,其声明的函数和变量可以在本模块或其它模块中使用。

(2)在C++中引用C语言中的函数和变量,在包含C语言头文件时,需要使用extern "C"来处理。 

1、extern表示函数和变量作用范围

HotSpot VM是一个由多文件组成的复杂系统,文件与文件之间难免会共享一些变量和函数,怎么办呢?例如在类加载时,为了处理并发问题会在多个文件中用到SystemDictionary_lock锁,这个锁在mutexLocker.hpp文件中被声明为外部变量,如下:

extern Monitor* SystemDictionary_lock; 

文件中用extern对SystemDictionary_lock做“外部变量声明”,在mutexLocker.cpp中定义了这个外部变量。在编译和连接时,系统会由此知道SystemDictionary_lock是一个已在别处定义的外部变量,并将在另一个文件中定义的外部变量的作用域扩展到本文件,在本文件中可以合法地引用外部变量SystemDictionary_lock。

2、兼容性处理

extern "C"在HotSpot VM中使用的比较多,如jni.h,如果你编写过native方法,那么这个头文件你应该熟悉,当写native方法的C或C++实现时,通常会引入这个头文件,这样我们就能在我们自己编写的函数中和虚拟机交互了。

可以借助javah工具生成我们需要的头文件,例如如下实例:

#include <jni.h>

#ifndef _Included_com_mprofiler_Test
#define _Included_com_mprofiler_Test

#ifdef __cplusplus
extern "C" {
#endif

JNIEXPORT void JNICALL Java_com_mprofiler_Test_helloWorld
  (JNIEnv *, jobject, jstring);

#ifdef __cplusplus
}
#endif

#endif
  

已经自动为我们引入了jni.h头文件。其中的宏__cplusplus是为了避免在C++中使用C语言。这样我们即可以通过C语言实现自己的native方法,也可以用C++实现自己的native方法。在jni.h头文件中同样有__cplusplus判断,也是为了支持用C或C++写native方法,不过由于C语言没有对象的概念,所以两者写起来还是有一些区别的,例如调用某个JNI函数,C语言写法:

(*env) -> GetStringUTFChars(env, str, NULL);

C++的写法如下:

env->GetStringUTFChars(jstr, nullptr);

另外还需要提示一点的是,通过如上的操作后,编译器会保持原本的名称。如果是C++函数,在编译的时候为了解决函数的多态问题,会将函数名和参数联合起来生成一个中间的函数名称。例如Launcher在启动HotSpot VM时,通常会通过dlsym()函数查找符号,如下:

ifn->CreateJavaVM = (CreateJavaVM_t)
            dlsym(libjvm, "JNI_CreateJavaVM");

这个JNI_CreateJavaVM就是通过extern "C"来保证函数本名的, 如果不使用,那么dlsym()可能无法在动态链接库libjvm.so中通过函数名找到这个函数。

还比如Async Profiler为了异步获取线程栈,要调用HotSpot VM内部的AsyncGetCallTrace()函数,由于这个函数有extern "C",所以使用函数本名查找即可。如果是C++,那么由于Name Mangling的存在,需要确定Name Mangling后的函数名称,如Async Profiler通过TLAB内部的函数统计分配速率时,用了_ZN11AllocTracer27send_allocation_in_new_tlab这样的函数名称。

 

本人最近准备出一个手写Hotspot VM的课程,超级硬核,从0开始写HotSpot VM,将HotSpot VM所有核心的实现全部走一遍,如感兴趣,加我速速入群。

群里可讨论虚拟机和Java性能剖析与故障诊断等话题,欢迎加入。

 

 

 

 

 

 

 

与C++的extern关键字在HotSpot VM中的重要应用相似的内容:

C++的extern关键字在HotSpot VM中的重要应用

extern关键字有两个用处: (1)extern在C/C++语言中表示函数和全局变量作用范围(可见性)的关键字,这个关键字会告诉编译器,其声明的函数和变量可以在本模块或其它模块中使用。 (2)在C++中引用C语言中的函数和变量,在包含C语言头文件时,需要使用extern "C"来处理。 1、ext

C#的多线程UI窗体控件显示方案 - 开源研究系列文章

上次编写了《LUAgent服务器端工具》这个应用,然后里面需要新启动一个线程去对文件进行上传到FTP服务器,但是新线程里无法对应用主线程UI的内容进行更改,所以就需要在线程里设置主UI线程里控件信息的方法,于是就有了此博文。此文记录的是一种高级用法。 为了实际的使用,笔者将线程操作放在独立的类当中,

C#的奇技淫巧:利用WinRM来远程操控其他服务器上的进程

前言:有时候远程服务器的进程你想偷偷去围观一下有哪些,或者对一些比较调皮的进程进行封杀,或者对一些自己研发的服务进行远程手动启动或者重启等,又不想打开远程桌面,只想悄咪咪地执行,那也许下面的文章会对你有启发。 前提条件 确保远程服务器(服务端)已启用WinRM。在远程服务器上运行以下命令可以启用和配

C#的关于窗体的类库方案 - 开源研究系列文章

这次想到了以前编写的关于应用的那个类库,不过当时的只是定义了显示接口,然后调用窗体显示。现在想到要把这个关于窗体的类库进行集合,统一调用,于是就把原来的代码进行了修改完善,终于得到了这次这个例子。 这个例子主要实现了4种关于窗体的形式。第1种为普通的显示文件的信息(即程序集信息里的那些信息);第2种

C#的基于.net framework的Dll模块编程(四) - 编程手把手系列文章

这次继续这个系列的介绍: 一、命名空间的起名; 对于C#来说,一般命名空间的建议是:公司名(或个人名称).产品名.分类名,比如我这边是用的这个:Lzhdim.LPF.Helper,意思是个人名Lzhdim,加上LPF为平台名,加上Helper分类为帮助类,其它的更长的请读者自己添加。 二、Dll库里

C#的基于.net framework的Dll模块编程(三) - 编程手把手系列文章

继续这个系列的博文: 一、设置DLL类库信息; 在接解决方案资源管理器中选择该Dll程序集项目,鼠标右键,选择属性,打开窗口。 点击“程序集信息”,打开并编辑该Dll程序集的相关信息; 二、代码折叠注释操作; 为了在编辑代码的时候让代码更加美观和专注性,需要将部分代码进行折叠,既做了注释,又能够将该

C#的基于.net framework的Dll模块编程(二) - 编程手把手系列文章

今天继续这个系列博文的编写。接上次的篇幅,这次介绍关于C#的Dll类库的创建的内容。因为是手把手系列,所以对于需要入门的朋友来说还是挺好的,下面开始咯: 一、新建Dll类库; 这里直接创建例子的Dll类库项目,至于项目文件目录的存放布局后面的例子中会介绍。 在解决方案资源管理器上鼠标右键,选择“添加

深度解读《深度探索C++对象模型》之默认构造函数

C++的默认构造函数的作用是什么?什么时候会需要一个默认构造函数,以及默认构造函数从哪里来?这篇文章将从编译器的角度来分析这个问题。

C++的动态分派在HotSpot VM中的重要应用

众所周知,多态是面向对象编程语言的重要特性,它允许基类的指针或引用指向派生类的对象,而在具体访问时实现方法的动态绑定。C++ 和 Java 作为当前最为流行的两种面向对象编程语言,其内部对于多态的支持对于单继承的实现非常类似。 首先来体现一下C++的动态分派,如下: class Base1{ pub

C#的重载决策

重载是许多编程语言支持的特性。所谓重载,就是指可以定义多个名称相同但参数(个数、类型和顺序)不同的方法(函数)。先来看一个例子: ```c# void Main() { char cvalue = 'a'; male m = new male(); m.write(cvalue); } class