linux环境下,java使用JNI调用c++方法

前言:先吐槽一下,网上写的真是乱七八糟,大部分都是复制粘贴,很多都跑不通,导致浪费了很多时间。吐槽结束,下面来说说我怎么写的(已跑通)

**

使用JNI的步骤如下:

**

1. 编写带有native声明的方法的java类
2. 使用javac命令编译所编写的java类
3. 然后使用javah + java类名生成扩展名为h的头文件
4. 使用C/C++实现本地方法
5. 将C/C++编写的文件生成动态连接库

用代码解释每一步:
第一步:编写带有native声明的方法的java类

public class HelloWorld {
	    public native void displayHelloWorld();//所有native关键词修饰的都是对本地的声明
	    static {
			   System.loadLibrary("hello");//载入本地库
		   }
            public static void main(String[] args) {
	              new HelloWorld().displayHelloWorld();
       	           }
}

注意:首先,你要用c++写的类,必须要用native关键字修饰,然后还需要用静态块加载到本地库,此处的hello与5.3必须一致,在Java中用System.loadLibrary()方法加载第5.3步产生的动态链接库文件,这个native()方法就可以在Java中被访问了。native 关键字告诉编译器(其实是JVM)调用的是该方法在外部定义

第二步:使用javac命令编译所编写的java类
这一步很简单:

javac HelloWorld.java

这一步没啥好说的

第三步:然后使用javah + java类名生成扩展名为h的头文件

javac HelloWorld.java -h JniH

这一步是比较坑的,因为新版jdk-10.0.2已经移除了javah命令工具,所有如果你在使用
javah HelloWorld 命令 将不会生成扩展名为h的头文件
生成的头文件如下:

/* DO NOT EDIT THIS FILE - it is machine generated */
#include 
/* Header for class HelloWorld */

#ifndef _Included_HelloWorld
#define _Included_HelloWorld
#ifdef __cplusplus
extern "C" {
#endif
/*
 * Class:     HelloWorld
 * Method:    displayHelloWorld
 * Signature: ()V
 */
JNIEXPORT void JNICALL Java_HelloWorld_displayHelloWorld
  (JNIEnv *, jobject);

#ifdef __cplusplus
}
#endif
#endif

第四步:使用C/C++实现头文件本地方法

#include "jni.h"
#include "HelloWorld.h"
 
//#include otherheaders
JNIEXPORT void JNICALL
Java_HelloWorld_displayHelloWorld(JNIEnv *env, jobject obj)
{
   printf("Helloworld!\n");
   return;
}

注意:这个方法名要和头文件里面的一样Java_HelloWorld_displayHelloWorld()

第五步:将C/C++编写的文件生成动态连接库
我自认为是最难的一步,也是我花时间最多的一步,坑也是最多的一步
使用 GCC 时 , 必须通知编译器在何处查找此 Java 本地方法的支持文件,不然就会报错,说找不到 "jni.h"文件
5.1 查找你的JDK的位置
以下,是一步接着一步查找的

yz@yaspeed:~/HelloWorld/JniH$ ls -lrt /etc/alternatives/java
lrwxrwxrwx 1 root root 43 12月 13 11:11 /etc/alternatives/java -> /usr/lib/jvm/java-11-openjdk-amd64/bin/java
yz@yaspeed:~/HelloWorld/JniH$ cd /usr/lib/jvm
yz@yaspeed:/usr/lib/jvm$ ls
default-java  java-1.11.0-openjdk-amd64  java-11-openjdk-amd64  java-1.8.0-openjdk-amd64  java-8-openjdk-amd64
yz@yaspeed:/usr/lib/jvm$ pwd
/usr/lib/jvm
yz@yaspeed:/usr/lib/jvm$ cd ./java-1.8.0-openjdk-amd64
yz@yaspeed:/usr/lib/jvm/java-1.8.0-openjdk-amd64$ ls
ASSEMBLY_EXCEPTION  bin  docs  include  jre  lib  man  src.zip  THIRD_PARTY_README
yz@yaspeed:/usr/lib/jvm/java-1.8.0-openjdk-amd64$ cd ./include
yz@yaspeed:/usr/lib/jvm/java-1.8.0-openjdk-amd64/include$ ls
classfile_constants.h  jdwpTransport.h  jni.h  jvmticmlr.h  jvmti.h  linux

5.2 将.c文件先编译成.o文件

gcc -I/usr/lib/jvm/java-1.8.0-openjdk-amd64/include -I/usr/lib/jvm/java-1.8.0-openjdk-amd64/include/linux -fPIC -c HelloWorldImpi.c

5.3 再将.o文件编译成.so文件(libhello.so lib后面必须加hello,因为在第一部的时候,家在本地的时候就已经定义)

gcc -shared -Wl,-soname,libhello.so.1 -o libhello.so.1.0 HelloWorldImpi.o

5.4 接下来将生成的共享库拷贝为标准文件名

cp libhello.so.1.0 libhello.so

5.5 最后通知动态链接程序此共享文件的路径。

export LD_LIBRARY_PATH=`pwd`:$LD_LIBRARY_PATH

第六步:测试

yz@yaspeed:~/HelloWorld$ java HelloWorld
Helloworld!

你可能感兴趣的:(linux)