JNI静态注册

Android 3XCN.com站长2023年6月27日 pm7:51发布6个月前更新
0
导航号,我的单页导航
目录

前言

之前其实写过,代码不见了,为了走一下流程,重新简单的写了一个。

PS:设计NDK环境配置这里不介绍哈

正文

静态注册

先由Java得到本地方法的声明,然后再通过JNI实现该声明方法。

  1. 优点: 理解和使用方式简单, 属于傻瓜式操作, 使用相关工具按流程操作就行, 出错率低

  2. 缺点: 当需要更改类名,包名或者方法时, 需要按照之前方法重新生成头文件, 灵活性不高

实战

定义Hello.java

加载Hello.so和native方法

package com.biumall.jni.one;
public class Hello {
    //1. load Hello.so
    static {
        System.loadLibrary("Hello");
    }
    // 2. define native hello()
    public static native String hello();
}
javac编译class

PS:javac对自定义对象不行,需要手动,可以看《JNI之类型介绍》和《JNI之自定义对象使用

通过javac编译Hello.class,然后javah编译

进入Hello.java目录

#当前位置:BiuJniStatic\src\main\java\com\biumall\jni\one
javac Hello.java

会在当前目录下生成

Hello.class
javah编译.h头文件

回退到 java目录

#当前位置:BiuJniStatic\src\main\java
javah com.biumall.jni.one.Hello

会在当前目录生成

com_biumall_jni_one_Hello.h
创建JNI

工程更目录创建JNI目录

BiuJniStatic/jni

把上面com_biumall_jni_one_Hello.h头文件拷贝到jni目录,并改名为

Hello.h

Hello.h拷贝一份为Hello.c

Hello.c也可以改为Hello.cpp,cpp中需要用C++写,否则会报错。

如果Hello.c中用了C++语法就会提示

error: member reference base type 'JNIEnv' (aka 'const struct JNINativeInterface *') is not a structure or union]

反过了Hello.cpp中用C语言写也会有类似的提示。

编写Hello.c

下面代码新增log打印,新增的活修改的都有备注

/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/* Header for class com_biumall_jni_one_Hello */

//----  add start -----
#include <android/log.h>
//----  add end   -----

#ifndef _Included_com_biumall_jni_one_Hello
#define _Included_com_biumall_jni_one_Hello
#ifdef __cplusplus
extern "C" {
#endif
/*
 * Class:     com_biumall_jni_one_Hello
 * Method:    hello
 * Signature: ()Ljava/lang/String;
 */
//----  modify start  -----

//LOG_TAG
#define LOG_TAG "from_static_jni_"
#define LOGD(...) __android_log_print(ANDROID_LOG_DEBUG, LOG_TAG, __VA_ARGS__)
#define LOGI(...) __android_log_print(ANDROID_LOG_INFO, LOG_TAG, __VA_ARGS__)
#define LOGE(...) __android_log_print(ANDROID_LOG_ERROR, LOG_TAG, __VA_ARGS__)

JNIEXPORT jstring JNICALL Java_com_biumall_jni_one_Hello_hello
  (JNIEnv * env , jclass jclazz){
    // [hello.c如果用C]
    LOGE("Java_com_biumall_jni_one_Hello_hello()");
    return (*env)->NewStringUTF(env, "Hello World !!!! --- from JNI ");
    // [hello.cpp如果用C++]
    // 如果hello.c中用了这个,无法build过,提示[error: member reference base type 'JNIEnv' (aka 'const struct JNINativeInterface *') is not a structure or union]
    //return env->NewStringUTF((char *)"Hello World !---from JNI");
};
//----  modify end   -----

#ifdef __cplusplus
}
#endif
#endif

PS : 上面---- 等备注的都是有更新的地方,其他的都是Hello.h一样。

配置Application.mk和Android.mk
Android.mk
LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)
LOCAL_LDLIBS := -llog
LOCAL_MODULE    := Hello
LOCAL_SRC_FILES := Hello.c
include $(BUILD_SHARED_LIBRARY)
  1. LOCAL_MODULE : 为生成so库的名字

  2. LOCAL_SRC_FILES : 为需要编译的源文件

Application.mk
APP_ABI := all

all表示全部平台,可编译出如下so库

├─arm64-v8a
├─armeabi
├─armeabi-v7a
├─mips
├─mips64
├─x86
└─x86_64

当然,可以指定编译指定平台的so。

# 指定生成armeabi-v7a
APP_ABI := armeabi-v7a
# 或者指定多个[多个用空格分开]
APP_ABI := armeabi-v7a arm64-v8a
编译JNI

进入

BiuJniStatic/jni

输入

ndk-build

如果ndk配置没问题,根目录会出现libs目录,也就生成对应的so库。

build.gradle配置so库
# BiuJniStatic/build.gradle
android {
    compileSdkVersion 31
    // 略
    sourceSets {
        main {
            jniLibs.srcDirs = ['libs']
        }
    }
}

编译运行,就可以测试Java调用JNI方法了。

参考文章

  1. [NDK开发]Android JNI 开发之静态注册

  2. [NDK开发]Android JNI 中新增JNI层日志打印

  3. NDK之使用第三方so库

版权声明 1、 本站名称: 91易搜
2、 本站网址: https://www.91es.com/
3、 本站文章: 部分来源于网络,仅供站长学习和参考,若侵权请留言
导航号,我的单页导航

暂无评论

评论审核已启用。您的评论可能需要一段时间后才能被显示。

暂无评论...