文档章节

Android JNI和NDK学习(03)--动态方式实现JNI

simpower
 simpower
发布于 07/28 18:13
字数 974
阅读 115
收藏 1

前面总结了静态实现JNI的方法,本文介绍如何动态实现JNI:JNI在加载时,会调用JNI_OnLoad,而卸载时会调用JNI_UnLoad,所以我们可以通过在JNI_OnLoad里面注册我们的native函数来实现JNI。下面就介绍该方法。

1 Android应用层代码

在eclipse中新建工程NdkLoad,工程文件NdkLoad.java的代码如下: 

复制代码
package com.skywang.ndk;

import android.app.Activity;
import android.os.Bundle;
import android.widget.TextView;
import android.util.Log;

public class NdkLoad extends Activity {
    public static final String TAG="skywang--NdkLoad";
    
    @Override
    public void onCreate(Bundle savedInstanceState)
    {
        super.onCreate(savedInstanceState);

        Log.d(TAG, "on create"); 
        TextView  myTextView = new TextView(this);
        myTextView.setText( HelloLoad() );
        setContentView(myTextView);
    }

    // jni中注册的方法
    public native String HelloLoad();

    static {
        // 加载本地libndk_load.so库文件
        System.loadLibrary("ndk_load");
    }
}
复制代码

public native String HelloLoad(); 这句话的作用是声明HelloLoad()这个本地方法。HelloLoad()是通过jni中注册到Android的方法,具体的实现在libndk_load.so中。
System.loadLibrary("ndk_load"); 这个函数的作用是加载libndk_load.so库文件。由于定义在NdkLoad类的static函数体中,所以在建立NdkLoad这个Acitivity时就会执行。

下面介绍ndk_load的具体实现。

我们知道,系统初始化JNI在加载时,会调用JNI_OnLoad(),而卸载时会调用JNI_UnLoad();所以,我们可以通过重写JNI_OnLoad(),在JNI_OnLoad()中将函数注册到Android中,以便能通过Java访问。在本文中,我们就是重写JNI_OnLoad()函数实现ndk_load库。

 

2 JNI动态注册的实现方法

2.1 编写JNI动态注册的方法

(01) 打开终端,切换到NdkLoad所在目录,新建jni目录。

假设NdkLoad所在目录为"/home/skywang/workspace/android_apps/NdkLoad",则执行以下命令:

$ cd /home/skywang/workspace/android_apps/NdkLoad/
$ mkdir jni

(02) 在jni目录下新建ndk_load.c,ndk_load.c的代码如下:

复制代码
#include <stdlib.h>
#include <string.h>
#include <stdio.h>
#include <jni.h>
#include <assert.h>


// 获取数组的大小
# define NELEM(x) ((int) (sizeof(x) / sizeof((x)[0])))
// 指定要注册的类,对应完整的java类名
#define JNIREG_CLASS "com/skywang/ndk/NdkLoad"


// 返回字符串"hello load jni"
JNIEXPORT jstring JNICALL native_hello(JNIEnv *env, jclass clazz)
{
    return (*env)->NewStringUTF(env, "hello load jni.");
}

// Java和JNI函数的绑定表
static JNINativeMethod method_table[] = {
    { "HelloLoad", "()Ljava/lang/String;", (void*)native_hello },//绑定
};

// 注册native方法到java中
static int registerNativeMethods(JNIEnv* env, const char* className,
        JNINativeMethod* gMethods, int numMethods)
{
    jclass clazz;
    clazz = (*env)->FindClass(env, className);
    if (clazz == NULL) {
        return JNI_FALSE;
    }
    if ((*env)->RegisterNatives(env, clazz, gMethods, numMethods) < 0) {
        return JNI_FALSE;
    }

    return JNI_TRUE;
}

int register_ndk_load(JNIEnv *env)
{
    // 调用注册方法
    return registerNativeMethods(env, JNIREG_CLASS,
            method_table, NELEM(method_table));
}

JNIEXPORT jint JNI_OnLoad(JavaVM* vm, void* reserved)
{
    JNIEnv* env = NULL;
    jint result = -1; 

    if ((*vm)->GetEnv(vm, (void**) &env, JNI_VERSION_1_4) != JNI_OK) {
        return result;
    }   

    register_ndk_load(env);

    // 返回jni的版本
    return JNI_VERSION_1_4;
}
复制代码

JNI_OnLoad()会在JNI注册时被调用。在JNI_OnLoad()中,调用register_ndk_load()。
register_ndk_load()调用registerNativeMethods()。
registerNativeMethods()中通过FindClass()找到class;然后通过RegisterNatives()将method_table注册到class中。method_table是JNINativeMethod类型。
JNINativeMethod的定义如下:

typedef struct {
    const char* name;      // Java中申明的Native函数名称
    const char* signature; // 描述了函数的参数和返回值
    void* fnPtr;           // 函数指针,指向C函数
} JNINativeMethod;

通过method_table,就将本地的native_hello()函数和注册到Java中的HelloLoad()绑定起来了。当我们在Java中调用HelloLoad()时,实际调用的是native_hello()。

(03) 在jni目录下新建Android.mk,Android.mk的代码如下:

复制代码
LOCAL_PATH := $(call my-dir)

include $(CLEAR_VARS)

LOCAL_MODULE    := ndk_load
LOCAL_SRC_FILES := ndk_load.c

include $(BUILD_SHARED_LIBRARY)

LOCAL_PATH := $(call my-dir)
复制代码

3 编译生成.so库文件

切换到NdkLoad工程目录,并执行ndk-build,生成.so库文件。执行的命令如下:

$ cd /home/skywang/workspace/android_apps/NdkLoad/
$ ndk-build

命令执行成功,则生成“libs/armeabi/libndk_load.so”库文件。若命令执行失败,请先确保已经导入了ndk环境变量(请参考“Android JNI和NDK学习(01)--搭建NDK开发环境”)!

4 执行工程

以下是在平板上运行的实际效果图:

 

点击下载:源代码

本文转载自:https://www.cnblogs.com/skywang12345/archive/2013/05/23/3092491.html

simpower
粉丝 27
博文 646
码字总数 46970
作品 0
海淀
程序员
私信 提问
Android JNI MAC OS环境配置

Android JNI MAC OS环境配置 http://whbzju.github.io/blog/2013/06/01/android-jni-config/ JUN 1ST, 2013 | COMMENTS 前言—JNI技术简介 JNI是Java Native Interface的缩写,即“Java本地调......

whb_zju
2013/06/02
2.2K
0
Android小白成长之初级篇:NDK配置

AndroidNDK(Native Development Kit)实际上属于JNI(Java Native Interface)的一部分,JNI就是允许Java代码和其他语言写的代码进行交互。Android基于Java平台,自然也是支持这个特征了,因...

76135
2012/07/15
1K
0
JNI和NDK的区别

NDK(Native Development Kit)“原生”也就是二进制 android常用的开发方式是java封装的库,而这些库的底层实现是由C/C++实现,如媒体,图形库等 java调用这样实现就需要用JNI(Java Native...

长平狐
2013/01/06
107
0
JNI和NDK的区别

NDK(Native Development Kit)“原生”也就是二进制 android常用的开发方式是java封装的库,而这些库的底层实现是由C/C++实现,如媒体,图形库等 java调用这样实现就需要用JNI(Java Native...

鉴客
2012/03/09
215
0
JNI和NDK的区别

NDK(Native Development Kit)“原生”也就是二进制 android常用的开发方式是java封装的库,而这些库的底层实现是由C/C++实现,如媒体,图形库等 java调用这样实现就需要用JNI(Java Native...

晨曦之光
2012/03/01
1K
0

没有更多内容

加载失败,请刷新页面

加载更多

计算机实现原理专题--二进制减法器(二)

在计算机实现原理专题--二进制减法器(一)中说明了基本原理,现准备说明如何来实现。 首先第一步255-b运算相当于对b进行按位取反,因此可将8个非门组成如下图的形式: 由于每次做减法时,我...

FAT_mt
今天
5
0
好程序员大数据学习路线分享函数+map映射+元祖

好程序员大数据学习路线分享函数+map映射+元祖,大数据各个平台上的语言实现 hadoop 由java实现,2003年至今,三大块:数据处理,数据存储,数据计算 存储: hbase --> 数据成表 处理: hive --> 数...

好程序员官方
今天
7
0
tabel 中含有复选框的列 数据理解

1、el-ui中实现某一列为复选框 实现多选非常简单: 手动添加一个el-table-column,设type属性为selction即可; 2、@selection-change事件:选项发生勾选状态变化时触发该事件 <el-table @sel...

everthing
今天
6
0
【技术分享】TestFlight测试的流程文档

上架基本需求资料 1、苹果开发者账号(如还没账号先申请-苹果开发者账号申请教程) 2、开发好的APP 通过本篇教程,可以学习到ios证书申请和打包ipa上传到appstoreconnect.apple.com进行TestF...

qtb999
今天
10
0
再见 Spring Boot 1.X,Spring Boot 2.X 走向舞台中心

2019年8月6日,Spring 官方在其博客宣布,Spring Boot 1.x 停止维护,Spring Boot 1.x 生命周期正式结束。 其实早在2018年7月30号,Spring 官方就已经在博客进行过预告,Spring Boot 1.X 将维...

Java技术剑
今天
18
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部