android 动态注册Jni函数

Java的Native方法的名称和底层的c/c++方法名称是通过java_<包名>_<类名>_<方法名>这种方式对应的,即它是静态注册的,例如在java层的Native 方法名为stringFromJNI,它的包名为hello,类名为 hellojni则在底层的 c函数名称为java_hello_hellojni_stringFromJNI,这样一来名称就变得很复杂,不便于阅读和维护,在Android NDK下面提供了动态注册这种方法,它让底层的函数命名没有这种局限性,在编写底层c 函数代码时更加灵活 下面是具体做法:



#include <jni.h>
#include <android/log.h>
#include <stdio.h>
#include <stdlib.h>
#include <config.h>
static JNIEnv *env=NULL;
static int nMethods=2;
const char* className="Example/PINVOKEMethod";
const char* dataClassName="Data/MyData";
extern "C"
{
	//接收上层传来的字节数组
	void jniReceiveData(JNIEnv *e, jclass c,jbyteArray bytes,jint length);
	//供上层获取解析后的数据
	void jniGetGPSData(JNIEnv *e, jclass c,jobject o);
};
//动态注册的方法
static JNINativeMethod gMethods[] ={
	{
		"ReceiveData","([BI)V",(void *)jniReceiveData
	},
	{
		"getData","(LData/MyData;)V",(void *)jniGetData
	},
};
void jniReceiveData(JNIEnv *e, jclass c,jbyteArray bytes,jint length)
{

}
void jniGetData(JNIEnv *e, jclass c,jobject o)
{
	
}
jint JNI_OnLoad(JavaVM* vm, void* reserved)
{
	jint result=-1;
	jclass clz=NULL;
	if(vm->GetEnv((void**)&env,JNI_VERSION_1_4)!=JNI_OK)
		return -1;
	clz=env->FindClass(className);
	if(env->RegisterNatives(clz,gMethods,nMethods)<0)
		return -1;
	return JNI_VERSION_1_4;
}

其中JNINativeMethod为一个结构体定义如下:


typedef struct {    
const char* name;      //Java中 native函数的名字  
const char* signature;  //用字符串描述的java native 函数的参数和返回值  
void* fnPtr;            //指向底层 C函数的函数指针  
} JNINativeMethod;


例如在上面中 ReceiveData 为java的native函数名称, 它对应的底层c函数名称为 jniReceiveData,其参数两个 第一个为byte[],第二个为int 类型。getData对应的为jniGetData,其参数为自定义类型包名为Data,数据类名为MyData。在Jni_OnLoad调用时进行一一对应,Jni_OnLoad为Ndk自带的函数,用户可以在里面进行一些初始化操作。




其中第二个参数具体定义方法如下:


具体的每一个字符的对应关系如下
字符 Java类型 C类型
V      void            void
Z       jboolean     boolean
I        jint              int
J       jlong            long
D      jdouble       double
F      jfloat            float
B      jbyte            byte
C      jchar           char
S      jshort          short


数组则以"["开始,用两个字符表示
[I       jintArray      int[]
[F     jfloatArray    float[]
[B     jbyteArray    byte[]
[C    jcharArray    char[]
[S    jshortArray   short[]
[D    jdoubleArray double[]
[J     jlongArray     long[]
[Z    jbooleanArray boolean[]


上面的都是基本类型。如果Java函数的参数是class,则以"L"开头,以";"结尾中间是用"/" 隔开的包及类名。而其对应的C函数名的参数则为jobject. 一个例外是String类,其对应的类为jstring
Ljava/lang/String; String jstring
Ljava/net/Socket; Socket jobject


例如:   ReceiveData 的具体定义则为: ReceiveData(byte[] data, int length);  getData 的具体定义为:(MyData data)


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值