我們知道一般在jni開發時,需要先根據java文件,利用javah生產對應的.h文件,再實現相應方法。運行時Java虛擬機加載相應運行庫,再逐一檢索,將java類中的本地方法和相應的JNI本地函數方法映射起來,但當本地接口方法很多時,會增加運行時間,降低運行效率。爲了解決這一問題,JNI機制提供了RegisterNatives() JNI函數,該函數允許JNI本地函數與java類中的本地方法直接映射在一起。
使用相應庫時,首先需要調用System.loadLibrary()加載相應庫,該方法執行過程中,會檢查JNI_OnLoad()函數是否被實現,因此想要直接註冊本地函數,需要在JNI_OnLoad()中調用RegisterNatives() 函數進行映射匹配。
假設存在一個java類,包含一個native接口
package com.zpm.Info;
public class ZPm {
static{
System.loadLibrary("zpm");
}
public native String get();
}
在jni目錄新建一個cpp文件,文件名隨便,這裏採用的是C++,採用C時,語法略有不同
/*
* Zpm.cpp
*
* Created on: 2018-7-3
* Author: z26442
*/
#include<jni.h>
#include<stdio.h>
jstring hi(JNIEnv* env, jobject obj){
return env->NewStringUTF("zhupumao");
}
const JNINativeMethod method[]={
{"get","()Ljava/lang/String;",reinterpret_cast<jstring*>(hi)}
};
JNIEXPORT jint JNICALL JNI_OnLoad(JavaVM *jvm,void *reserved){
JNIEnv *env;
if(jvm->GetEnv((void**)&env,JNI_VERSION_1_4)!=JNI_OK){
return -1;
}
jclass clz=env->FindClass("com/zpm/Info/ZPm");
env->RegisterNatives(clz,method,sizeof(method)/sizeof(method[0]));
return JNI_VERSION_1_4;
}
新建Android.mk文件
LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)
LOCAL_MODULE := zpm
LOCAL_SRC_FILES := Zpm.cpp
include $(BUILD_SHARED_LIBRARY)
編譯生產.so庫。這樣就可以使用了。