《JVM第6课》本地方法栈

news/2024/10/31 14:38:38/

文章目录

    • 1 什么是本地方法
      • 1.1 本地方法的好处
      • 1.2 声明本地方法
      • 1.3 实现本地方法
        • 1. 生成头文件
        • 2. 编写C语言实现
        • 3. 编译C代码
        • 4. 运行Java程序
      • 1.4 使用JNA
      • 1.5 总结
    • 2 本地方法栈

无痛快速学习入门JVM,欢迎订阅本免费专栏

1 什么是本地方法

首先要知道什么是本地方法,本地方法并不是 JVM 自己的方法,也不是 jre 里面的方法,而是指那些操作系统自己的方法(如C/C++方法),它们在操作系统目录里。可以这么理解,本地方法就是计算机操作系统对外提供的方法,JVM 通过调用这些方法可以实现 Java 程序和计算机的交互。

1.1 本地方法的好处

  1. 访问操作系统资源:直接调用操作系统的API,例如文件系统、网络接口、图形用户界面等。
  2. 性能优化:对于某些计算密集型任务,使用C或C++等语言实现可以显著提高性能。
  3. 使用现有库:利用已经存在的C/C++库,避免重复开发和维护。
  4. 硬件访问:直接访问硬件设备,例如摄像头、传感器等。

所以如果我们想优化计算密集型任务的性能,或是调用 Java 中没有实现的计算机功能,我们可以自己实现一个本地方法。

1.2 声明本地方法

在Java中,本地方法通过native关键字声明。例如:

java">public class MyClass {// 声明本地方法public native void nativeMethod();// 静态块中加载本地库static {System.loadLibrary("mylib"); // 加载名为mylib的本地库}public static void main(String[] args) {new MyClass().nativeMethod(); // 调用本地方法}
}

1.3 实现本地方法

本地方法的实现通常使用JNI(Java Native Interface)或JNA(Java Native Access)来完成。以下是使用JNI实现本地方法的步骤:

1. 生成头文件

使用javah工具生成包含本地方法签名的C头文件。假设上面的Java类保存为MyClass.java,编译后生成MyClass.class,然后运行:

javah -jni MyClass

这将生成一个名为MyClass.h的头文件,内容如下:

/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/* Header for class MyClass */#ifndef _Included_MyClass
#define _Included_MyClass
#ifdef __cplusplus
extern "C" {
#endif
/** Class:     MyClass* Method:    nativeMethod* Signature: ()V*/
JNIEXPORT void JNICALL Java_MyClass_nativeMethod(JNIEnv *, jobject);#ifdef __cplusplus
}
#endif
#endif
2. 编写C语言实现

根据生成的头文件,编写C语言实现。例如:

#include <jni.h>
#include <stdio.h>
#include "MyClass.h"// 实现本地方法
JNIEXPORT void JNICALL Java_MyClass_nativeMethod(JNIEnv *env, jobject obj) {printf("Hello from native method!\n");
}
3. 编译C代码

将C代码编译成动态链接库。假设C文件名为mylib.c,编译命令如下:

在Linux上:

gcc -shared -o libmylib.so -I/usr/lib/jvm/java-8-openjdk-amd64/include -I/usr/lib/jvm/java-8-openjdk-amd64/include/linux mylib.c

在Windows上:

cl -I"%JAVA_HOME%\include" -I"%JAVA_HOME%\include\win32" -LD mylib.c -Fe mylib.dll
4. 运行Java程序

确保动态链接库在Java程序的库路径中,然后运行Java程序:

java -Djava.library.path=. MyClass

1.4 使用JNA

JNA是一种更简单的方式来调用本地库,不需要编写C代码。以下是一个使用JNA的示例:

  1. 添加JNA依赖:在项目中添加JNA的依赖。如果你使用Maven,可以在pom.xml中添加:
java"><dependency><groupId>net.java.dev.jna</groupId><artifactId>jna</artifactId><version>5.8.0</version>
</dependency>
  1. 定义接口:定义一个接口来映射本地库中的函数。
import com.sun.jna.Library;
import com.sun.jna.Native;public interface MyLib extends Library {MyLib INSTANCE = Native.load("mylib", MyLib.class);void nativeMethod();
}
  1. 调用本地方法
java">public class MyClass {public static void main(String[] args) {MyLib.INSTANCE.nativeMethod(); // 调用本地方法}
}

1.5 总结

本地方法是Java程序中的一种特殊方法,其声明在Java代码中,但实现由非Java语言编写。通过本地方法,Java程序可以访问操作系统资源、优化性能、使用现有库和直接访问硬件设备。常见的实现方式包括JNI和JNA。

2 本地方法栈

2.1 特点

  1. 线程私有:每个线程都有自己的本地方法栈,与Java虚拟机栈一样,本地方法栈也是线程私有的。
  2. 存储结构本地方法栈中的每个栈帧(Frame)对应一次本地方法的调用。栈帧中包含本地方法的参数、局部变量、操作数栈等信息。
  3. 调用机制:当Java代码调用一个本地方法时,JVM会创建一个新的栈帧并将其压入本地方法栈。本地方法执行完毕后,栈帧会被弹出并丢弃。

2.2 本地方法栈与Java虚拟机栈的区别

  • 用途不同:Java虚拟机栈用于支持Java方法的执行,而本地方法栈用于支持本地方法的执行。
  • 实现方式:Java虚拟机栈的实现由JVM规范规定,而本地方法栈的实现通常依赖于具体的JVM实现和操作系统的ABI(Application Binary Interface)。
  • 数据类型:Java虚拟机栈主要处理Java类型的值,而本地方法栈可能涉及更广泛的C/C++类型或其他原生类型。

2.3 本地方法栈的工作流程

  1. 方法调用:当Java代码中调用一个声明为native的方法时,JVM会查找该方法的本地实现。
  2. 栈帧创建:JVM为本地方法创建一个新的栈帧,并将其压入本地方法栈
  3. 参数传递:调用本地方法所需的参数会被从Java虚拟机栈复制到本地方法栈的栈帧中。
  4. 方法执行:本地方法在本地方法栈中执行,可以访问操作系统资源、硬件设备等。
  5. 结果返回:本地方法执行完毕后,结果会被从本地方法栈复制回Java虚拟机栈,然后继续执行Java代码。
  6. 栈帧弹出本地方法栈中的栈帧被弹出并丢弃。

2.4 总结

本地方法栈是JVM中用于支持本地方法调用的重要数据结构。通过本地方法栈,Java程序可以调用用其他语言编写的代码,从而实现更广泛的功能和更高的性能。


http://www.ppmy.cn/news/1543360.html

相关文章

T527 Android13开发环境

一、环境系统选择&#xff1a; 开发环境建议选用ubuntu&#xff0c;下面ubuntu18.04 为例 二、环境依赖安装&#xff1a; 1、安装依赖库 sudo apt-get install -y git-core gnupg flex bison gperf build-essential zip curl zlib1g-dev gcc-multilib gmultilib libc6-dev-…

Linux:编辑器Vim和Makefile

✨✨所属专栏&#xff1a;Linux✨✨ ✨✨作者主页&#xff1a;嶔某✨✨ vim的三种常用模式 分别是命令模式&#xff08;command mode&#xff09;、插入模式&#xff08;Insert mode&#xff09;和底行模式&#xff08;last line mode&#xff09; 各模式的功能区分如下&…

【PythonWeb开发】Flask-RESTful字段格式化

字段格式化的意思就是对api接口返回的响应做出规范。 具体的思想是&#xff0c;先定义一个格式化输出样板fields&#xff0c;然后在类视图中的视图方法上加marshal_with装饰器使得格式化生效。这样做的有什么特别的用处吗&#xff1f;特性&#xff1a; 显示出我们设计的数据结构…

背包九讲——树形背包问题(有依赖的背包)

目录 树形背包问题 问题引入&#xff1a; 问题解读&#xff1a; 算法例题&#xff1a;10. 有依赖的背包问题 - AcWing题库 题目&#xff1a; 算法实现&#xff1a; 代码实现&#xff1a; 背包问题第七讲——树形背包问题&#xff08;有依赖的背包&#xff09; 背包问题是…

Docker学习笔记【从入门到精通】

目录 一、Docker是什么 二、Docker容器部署项目 三、Docker的核心特点 四、Docker的应用场景和使用方式 五、Docker安装部署 1.Docker引擎 2.Docker组成 &#xff08;1&#xff09;Docker Daemon &#xff08;2&#xff09;Rest接口 &#xff08;3&#xff09;Docker …

Dockerfile制作Oracle19c镜像

Dockerfile文件 cat > Dockerfile << EOF # 使用 Oracle Linux 8 作为基础镜像 FROM oraclelinux:8# 复制 Oracle 19c 安装包 COPY oracle-database-ee-19c-1.0-1.x86_64.rpm /tmp/# 安装 Oracle 19c 数据库和依赖 RUN yum localinstall -y /tmp/oracle-database-ee-…

C/C++内存和内存管理

一.C/C内存分布 首先来检测一下对于内存的分布知识是否理解通透了&#xff0c;请看下面代码和回答相关问题。 int globalVar 1; static int staticGlobalVar 1; void Test() {static int staticVar 1;int localVar 1;int num1[10] { 1, 2, 3, 4 };char char2[] "a…

【Linux】Kafka部署

1、部署 # 解压 [appuserlocalhost app]$ tar -zxvf kafka_2.13-3.0.0.tgz# 修改配置文件&#xff08;具体配置见下文&#xff09; [appuserlocalhost app]$ cd kafka_2.13-3.0.0 [appuserlocalhost kafka_2.13-3.0.0]$ vim config/server.properties # 在所有集群节点启动Kaf…