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