Linux 下 JNI 实现 Java 调用 c的例子
环境: Ubuntu16_x64 环境, cmake
下面提供一个简单的 Demo,便于理解JNI的特性。
编写 java 程序
JNIDemo.java
public class JNIDemo
{
static
{
/* 1 加载 lib 动态库 */
System.loadLibrary("native");
}
/* 2 C实现 Java和 C的映射关系
* 2.1 JNIEXPORT jint JNICALL JNI_OnLoad(JavaVM *vm, void *reserved); // 注册
* 2.2 JNIEXPORT void JNICALL JNI_OnUnload(JavaVM *vm, void *reserved); // 注销
* 2.3 java 实现本地接口函数 public native int hello(int a);
*/
public native int hello(int a);
public static void main(String args[])
{
JNIDemo d = new JNIDemo();
d.hello(2);
}
}
编写 c 库
CMakeLists.txt
CMAKE_MINIMUM_REQUIRED(VERSION 3.0)
PROJECT(Hello) # 指定项目名
# 添加头文件路径
SET(INC_PATH /usr/local/java/jdk1.8.0_192/include
/usr/local/java/jdk1.8.0_192/include/linux/)
INCLUDE_DIRECTORIES(${INC_PATH})
# 添加连接库路径
#SET(LIB_PATH )
#LINK_DIRECTORIES(${LIB_PATH})
# 添加源文件
SET(SRC_LIST ${PROJECT_SOURCE_DIR}/src/native.c)
#SET(CMAKE_C_FLAGS )
#SET(CMAKE_CXX_FLAGS )
#ADD_EXECUTABLE(${PROJECT_NAME} ${SRC_LIST})
ADD_LIBRARY(native SHARED ${SRC_LIST})
编写 native.c
#include <jni.h>
#include <stdio.h>
JNIEXPORT jint JNICALL
hello(JNIEnv *env, jobject cls, jint m)
{
printf("Hello, java--c!\n");
return 0;
}
/* JNI 方法列表 */
static const JNINativeMethod methods[] =
{
{"hello", "(I)I", (void *)hello},
};
/*
* JNIEnv 是一个与线程相关的变量,不同线程的JNIEnv彼此独立。
* JavaVM 是虚拟机在JNI层的代表,在一个虚拟机进程中只有一个JavaVM。
*/
JNIEXPORT jint JNICALL
JNI_OnLoad(JavaVM *vm, void *reserved)
{
JNIEnv *env = NULL;
jclass cls;
/* 第一步,获取 Java 运行环境 */
if((*vm)->GetEnv(vm, (void **)&env, JNI_VERSION_1_4))
{
return JNI_ERR;
}
/* 找到对应的类名 */
cls = (*env)->FindClass(env, "JNIDemo");
if(cls == NULL)
{
return JNI_ERR;
}
/* 注册本地函数 map java <--> c */
if((*env)->RegisterNatives(env, cls, methods, 1) < 0)
{
return JNI_ERR;
}
return JNI_VERSION_1_4;
}
JNIEXPORT void JNICALL
JNI_OnUnload(JavaVM *vm, void *reserved)
{
}
执行命令如下
$cmake ../
$make
$export LD_LIBRARY_PATH=./
运行
注意c和cpp 调用 JNI 注册方法的差异。
感兴趣的朋友可以对比 jdk_1.6.0_43/include/jni.h 中的C语言函数声明和 CPP成员函数声明的区别。