`
mfcai
  • 浏览: 404115 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

深入了解android平台的jni---注册native函数

阅读更多
注册native函数有两种方法:静态注册和动态注册。
1、静态注册方法
根据函数名找到对应的JNI函数:Java层调用函数时,会从对应的JNI中寻找该函数,如果没有就会报错,如果存在则会建立一个关联联系,以后在调用时会直接使用这个函数,这部分的操作由虚拟机完成。
静态方法就是根据函数名来遍历java和jni函数之间的关联,而且要求jni层函数的名字必须遵循
特定的格式,其缺点在于:
1)javah生成的jni层函数特别长;
2)初次调用native函数时要根据名字搜索对应的jni层函数来建立关联联系,这样影响效率。

[b]2、动态注册方法[/b]
JNI 允许你提供一个函数映射表,注册给Jave虚拟机,这样Jvm就可以用函数映射表来调用相应的函数,
就可以不必通过函数名来查找需要调用的函数了。
Java与JNI通过JNINativeMethod的结构来建立联系,它在jni.h中被定义,其结构内容如下:
typedef struct {
const char* name;
const char* signature;
void* fnPtr;
} JNINativeMethod;
第一个变量name是Java中函数的名字。
第二个变量signature,用字符串是描述了函数的参数和返回值
第三个变量fnPtr是函数指针,指向C函数。
当java通过System.loadLibrary加载完JNI动态库后,紧接着会查找一个JNI_OnLoad的函数,如果有,就调用它,
而动态注册的工作就是在这里完成的。
1)JNI_OnLoad()函数
JNI_OnLoad()函数在VM执行System.loadLibrary(xxx)函数时被调用,它有两个重要的作用:
指定JNI版本:告诉VM该组件使用那一个JNI版本(若未提供JNI_OnLoad()函数,VM会默认该使用最老的JNI 1.1版),如果要使用新版本的JNI,
例如JNI 1.4版,则必须由JNI_OnLoad()函数返回常量JNI_VERSION_1_4(该常量定义在jni.h中) 来告知VM。
初始化设定,当VM执行到System.loadLibrary()函数时,会立即先呼叫JNI_OnLoad()方法,因此在该方法中进行各种资源的初始化操作最为恰当,
2)RegisterNatives
RegisterNatives在AndroidRunTime里定义
syntax:
jint RegisterNatives(jclass clazz, const JNINativeMethod* methods,jint nMethods)


3、在android中加入自定义的native函数
JNI在Android层次结构中的作用如下图所示:
深入了解android平台的jni---注册native函数
  
在Android中,主要的JNI代码在以下的路径中:
Android源码根目录/frameworks/base/core/jni/
这个路径中的内容将被编译成库libandroid_runtime.so,这就是一个普通的动态库,被放置在目标系统的/system/lib目录中.除此之外,Android还包含其他的JNI库,例如,媒体部分的JNI目录frameworks/base/media/jni/中,被编译成库libmedia_jni.so.
JNI中的各个文件实际上就是C++的普通文件,其命名一般和支持的Java类有对应关系。
这种关系是习惯上的写法,而不是强制的。

1)注册JNI方法
在Android源码根目录/frameworks/base/services/jni/目录下有一个onload.cpp文件,加入 jni函数申明和jni函数注册方法
#include "JNIHelp.h"   
#include "jni.h"   
#include "utils/Log.h"   
#include "utils/misc.h"   
  
namespace android {  
int register_android_server_AlarmManagerService(JNIEnv* env);  
int register_android_server_BatteryService(JNIEnv* env);  
int register_android_server_InputApplicationHandle(JNIEnv* env);  
int register_android_server_InputWindowHandle(JNIEnv* env);  
int register_android_server_InputManager(JNIEnv* env);  
int register_android_server_LightsService(JNIEnv* env);  
int register_android_server_PowerManagerService(JNIEnv* env);  
int register_android_server_UsbDeviceManager(JNIEnv* env);  
int register_android_server_UsbHostManager(JNIEnv* env);  
int register_android_server_VibratorService(JNIEnv* env);  
int register_android_server_SystemServer(JNIEnv* env);  
int register_android_server_location_GpsLocationProvider(JNIEnv* env);  
int register_android_server_connectivity_Vpn(JNIEnv* env);  
int register_android_server_HelloService(JNIEnv *env);  //此处加入自定义jni函数申明
};  
  
using namespace android;  
  
extern "C" jint JNI_OnLoad(JavaVM* vm, void* reserved)  
{  
    JNIEnv* env = NULL;  
    jint result = -1;  
  
    if (vm->GetEnv((void**) &env, JNI_VERSION_1_4) != JNI_OK) {  
        LOGE("GetEnv failed!");  
        return result;  
    }  
    LOG_ASSERT(env, "Could not retrieve the env!");  
  
    register_android_server_PowerManagerService(env);  
    register_android_server_InputApplicationHandle(env);  
    register_android_server_InputWindowHandle(env);  
    register_android_server_InputManager(env);  
    register_android_server_LightsService(env);  
    register_android_server_AlarmManagerService(env);  
    register_android_server_BatteryService(env);  
    register_android_server_UsbDeviceManager(env);  
    register_android_server_UsbHostManager(env);  
    register_android_server_VibratorService(env);  
    register_android_server_SystemServer(env);  
    register_android_server_location_GpsLocationProvider(env);  
    register_android_server_connectivity_Vpn(env);  
    register_android_server_HelloService(env); //jni方法注册
  
    return JNI_VERSION_1_4;  
}  

onload.cpp文件上部分为注册函数的声明,下部分为调用各种注册函数,而这些注册函数就是JNI方法的注册函数! 正是通过这些注册函数,上层才能调用注册的JNI方法.
以register_android_server_HelloService为例,来看一个注册函数的具体实现过程是如何的。
打开com_android_service_HelloService.cpp文件

2)加入注册函数的实现代码,如下:
int register_android_server_HelloService(JNIEnv *env) { 
        return jniRegisterNativeMethods(env, "com/android/server/HelloService", method_table, NELEM(method_table)); 

#其中jniRegisterNativeMethods为注册JNI方法函数,
#此函数的第二个参数为对应着java类即HelloService.java的文件名,第三个参数为注册的方法表

3)加入jni方法表

static const JNINativeMethod method_table[] = { 
    {"init_native", "()Z", (void*)hello_init}, 
    {"setVal_native", "(I)V", (void*)hello_setVal}, 
    {"getVal_native", "()I", (void*)hello_getVal}, 
}; 
4)方法表内各个接口的实现代码
static void hello_setVal(JNIEnv* env, jobject clazz, jint value) { 
     val = value; 
    LOGI("Hello JNI: set value %d to device.", val); 

static jint hello_getVal(JNIEnv* env, jobject clazz) { 
        LOGI("Hello JNI: get value %d from device.", val); 
        return val; 
    }
static jboolean hello_init(JNIEnv* env, jclass clazz) { 
        LOGI("Hello JNI: initializing......"); 
        return -1;       
    }
   
完整代码如下:
namespace android
{
  int val;
  static void hello_setVal(JNIEnv* env, jobject clazz, jint value) {  
     val = value;  
    LOGI("Hello JNI: set value %d to device.", val);  
  }  
  static jint hello_getVal(JNIEnv* env, jobject clazz) {  
        LOGI("Hello JNI: get value %d from device.", val);  
        return val;  
    } 
 static jboolean hello_init(JNIEnv* env, jclass clazz) {  
        LOGI("Hello JNI: initializing......");  
        return -1;        
    } 
  static const JNINativeMethod method_table[] = {  
    {"init_native", "()Z", (void*)hello_init},  
    {"setVal_native", "(I)V", (void*)hello_setVal},  
    {"getVal_native", "()I", (void*)hello_getVal},  
   }; 
   int register_android_server_HelloService(JNIEnv *env) {  
        return jniRegisterNativeMethods(env, "com/android/server/HelloService", method_table, NELEM(method_table));  
   }  
}

本文欢迎转载,转载请注明出处与作者
出处:http://blog.sina.com.cn/staratsky
作者:流星
分享到:
评论

相关推荐

    jni-example.zip

    Android jni基础知识, # This is a test examlpe foe android jni 1、实现一个android工程,在其Android.mk文件中不要定义jni相关的东西,代码中可以尽情的使用 2、使用mmm 命令编译该工程,能够编译成功的。...

    JNI入门Demo3.0

    JNI入门Demo3.0,用于演示JNI中是如何动态注册Native函数。

    JniCallback.zip_Android jni_android_jni android_jni callback_jni

    实现了android 应用层和通过JNI层回调函数完成应用与native 本地代码之间的相互访问,对于了解JNI回调函数的用法有帮助

    Android JNI多线程编程回调JAVA函数

    通过JNI接口静态注册的native方法去创建线程,同时提供native回调Java的方法。通过这个框架可以去实现线程监听某一个状态,然后回调Java的方法(如发消息去通知顶层,实现显示)

    深入理解Android:卷I--详细书签版

    2.3.2 Java的native函数和总结 17 2.4 JNI层MediaScanner的分析 17 2.4.1 注册JNI函数 18 2.4.2 数据类型转换 22 2.4.3 JNIEnv介绍 24 2.4.4 通过JNIEnv操作jobject 25 2.4.5 jstring介绍 27 2.4.6 JNI类型...

    JNI之Hello-JNI进阶

    2. 编写HelloJni工程,在主Activity(本例:HelloJni.java)里声明native函数: 如下: public native String stringFromJNI(); public native double add(double a, double b); public native double sub...

    《深入理解Android》卷Ⅰ

    2.3.2 Java的native函数和总结 2.4 JNI层MediaScanner的分析 2.4.1 注册JNI函数 2.4.2 数据类型转换 2.4.3 JNIEnv介绍 2.4.4 通过JNIEnv操作jobject 2.4.5 jstring介绍 2.4.6 JNI类型签名介绍 2.4.7 垃圾回收 2.4.8 ...

    hello-jni.rar_android_sdk hello-jni

    不过既然整个Android平台是开源的,我们可以通过Google发布的源代码来找到一些线索(比如frameworks/base/media/jni/目录),依葫芦画瓢的实现上层JAVA程序通过JNI来调用Native C程序中的函数。

    Android系统jni编程和调用实例

    Android开发通过jni技术编写C语言方法并实现...自己用C语言编写了一个动态链接库ndkfoo,并通过java调用动态库中的函数invokeNativeFunction,返回一个字符串,在android程序中显示出来。是初学jni开发方法的简单例子。

    Android NDK+JNI 练习1

    Native C函数调用Java函数,如何活用JNI和 NDK 来让 Native C函数顺利呼叫Java函数

    《深度理解Android:第一卷》

    2.3.2 Java的native函数和总结 / 17 2.4 JNI层MediaScanner的分析 / 17 2.4.1 注册JNI函数 / 18 2.4.2 数据类型转换 / 22 2.4.3 JNIEnv介绍 / 24 2.4.4 通过JNIEnv操作jobject / 25 2.4.5 jstring介绍 / 27 ...

    Android代码-Android

    Android 本目录下的代码都是来自于我的...JniCallback 演示了如何从Native线程回调Java的函数,相关博文:Android开发实践:JNI层线程回调Java函数示例 JniBuffer 演示了各种从Java端到Native层的Buffer传递方法,相

    深入理解Android卷1全

    2.3.2 Java的native函数和总结 / 17 2.4 JNI层MediaScanner的分析 / 17 2.4.1 注册JNI函数 / 18 2.4.2 数据类型转换 / 22 2.4.3 JNIEnv介绍 / 24 2.4.4 通过JNIEnv操作jobject / 25 2.4.5 jstring介绍 / 27 2.4.6 ...

    JNI完全技术手册 带完整书签

    3、实例三、在jni函数中访问java类中的对象实例域... 58 4、实例四:在jni函数中访问类的静态实例域... 60 5、实例五:在jni函数中调用java对象的方法... 60 6、实例六:在jni函数中调用java类的静态方法... 61 ...

    androidnative.pri:在不使用JNI的情况下从Qt调用Android函数

    在不使用JNI的情况下从Qt调用Android函数 它是项目的一个分支,旨在提供一个无需使用JNI即可从Qt / QML访问Android功能的库。 备注:该项目仅支持gradle构建系统。 特征 SystemDispatcher-C ++ / Qt和Java / ...

    AndroidNDKSimple

    //-----c调用java成员函数--------- private native void AccessStaticMethodDemo(Dummy p); private native void AccessInstaceMethodDemo(Dummy p); //------c调用java数据成员------------ private int ...

    Android JNI使用demo源码

    JNI使用demo,包含C/C++调用Native和jNativeC/C++ Cmake文件的使用和配置规则,注册规则 JNI函数的动态注册和动态注册示例 同时附上JNI文章详细解说

    static_jni.zip

    在Java虚拟机加载so库时,如果发现含有上面两个宏定义的函数时就会链接到对应Java层的native方法,那么怎么知道对应Java中的哪个类的哪个native方法呢,我们仔细观察JNI函数名的构成其实是:以Java为前缀,并且用“_...

    jni回调Java层函数示例

    NI是Java Native Interface的缩写,是Java平台的重要特性,使得Java代码可以方便地与C/C++代码...本文主要给出一份示例代码(工程文件见附件),描述如何在Android的JNI层开启一个线程,并在线程中回调Java层的函数。

    JNI函数字节码

    ENV的方法可装为so中的字节码

Global site tag (gtag.js) - Google Analytics