博主的测试环境为:
mac os
其他操作系统其实差不多
步骤一
编写一个java类:
package com.test;
public class MyNative {
public static void main(String[] args) {
System.out.println(func_int());
}
static{
/**
这里填写动态库的全路径即可
mac环境使用mynative.jnilib
linux环境使用mynative.so
windows环境使用mynative.dll
**/
System.load("/Users/mynative.jnilib");
}
public static native int func_int();
}
步骤二
- 将上述java文件编译为MyNative.class文件
- 打开命令行cd到MyNative.class所在的文件夹
- 然后执行命令
javah -jni -encoding utf-8 com.test.MyNative
- 然后就能获得头文件"com_test_MyNative.h"
- 在当前文件夹新建一个cpp文件,命名随意,假设命名为"com_test_MyNative.cpp"
经过一番操作后,如无意外,你会获得类似下图文件:
步骤三
正式开撸C++代码
#include "com_test_MyNative.h"
JNIEXPORT jint JNICALL Java_com_test_MyNative_1func_1int(JNIEnv *env, jobject)
{
jint a = 111;
return a;
}
为简洁教学,就简单几句代码
步骤四
编译C++
执行下面命令
g++
-I "/Library/Java/JavaVirtualMachines/jdk1.8.0_311.jdk/Contents/Home/include"
-I "/Library/Java/JavaVirtualMachines/jdk1.8.0_311.jdk/Contents/Home/include/darwin"
-shared
-o mynative.jnilib com_test_MyNative.cpp
命令解释
-o 不解释
-I 参数是用来指定头文件所在目录,不加上的话,会报错找不到’jni.h’ file not found
-shared 表示编译为动态库,不加上的,否则会报错如下
Undefined symbols for architecture x86_64:
“_main”, referenced from:
implicit entry/start for main executable
ld: symbol(s) not found for architecture x86_64
clang: error: linker command failed with exit code 1 (use -v to see invocation)
到此,如无意外,你的文件将会类似如下图
步骤五
直接运行MyNative.java文件,如无意外,会打印出"111"
OK! java调用jni教程至此结束!接下来开始C++调用java
步骤一
修改MyNative.java,添加一个静态java方法,以待C++调用
package com.test;
public class MyNative {
public static void main(String[] args) {
System.out.println(func_int());
}
static{
/**
这里填写动态库的全路径即可
mac环境使用mynative.jnilib
linux环境使用mynative.so
windows环境使用mynative.dll
**/
System.load("/Users/mynative.jnilib");
}
public static native int func_int();
// 添加一个静态方法
public static String bigunOK(String ss){
System.out.println("C++调用java:"+ss);
return "ok"+ss;
};}
步骤二
在c++中调用上述的bigunOK方法
#include "com_test_MyNative.h"
jclass cls; // 注意!这里必须要用全局变量,否则运行失败
jmethodID mid; // 注意!这里必须要用全局变量,否则运行失败
JNIEXPORT jint JNICALL Java_com_test_MyNative_1func_1int(JNIEnv *env, jobject)
{
// c++调用java --- start
cls = (env)->FindClass("com/bigun/packtool/PackNative");
if (cls != 0)
{
// 获取方法ID, 通过方法名和签名, 调用静态方法
mid = (env)->GetStaticMethodID(cls, "bigunOK", "(Ljava/lang/String;)Ljava/lang/String;");
// 此外,如果静态方法是没有参数的,那么填 "()V",如(env)->GetStaticMethodID(cls, "bigunOK", "()V");
}
if (mid != 0)
{
const char *name = "World";
jstring arg = (env)->NewStringUTF(name);
jstring result = (jstring)(env)->CallStaticObjectMethod(cls, mid, arg);
const char *str = (env)->GetStringUTFChars(result, 0);
printf("Result of sayHello: %s\n", str);
(env)->ReleaseStringUTFChars(result, 0);
}
// c++调用java --- end
jint a = 111;
return a;
}
OK~!重新编译一下,然后运行java代码,如无意外,则输入如下图所示:
恭喜你!成功啦~!