美文网首页
2. 【干货】火爆全网的《超全NDK精品教程》JNI 基础 (i

2. 【干货】火爆全网的《超全NDK精品教程》JNI 基础 (i

作者: 鹏城十八少 | 来源:发表于2023-03-12 19:14 被阅读0次

C和c++有一些小区别

C和c++有一些小区别

https://juejin.cn/post/7039948455259111461

C++条件编译:#ifdef

ifndef : 头文件引用, 可用通过大标记的方式! 否则会出现循环引入! 

// 打个标记,防止反复引入 copy 内容#ifndef_Included_com_darren_ndk12_NdkTest#define_Included_com_darren_ndk12_NdkTest#ifdef__cplusplus// 如果是 c++ 则统一用 C 的编译方式// 会指示编译器这部分代码按C语言的进行编译,而不是C++的。// C语言并不支持函数重载,因此编译C语言代码的函数时不会带上函数的参数类型,一般只包括函数名。

extern “C”

指定以"C"的方式来实现native函数,当然你也可以选择用extern "C++"。两种方式大致一样,主要是对env的操作方式略有区别

其中extern “C”根据需要动态添加,如果是C++代码,则必须要添加extern “C”声明,如果是C代码,则不用添加

JNIEXPORT

宏定义,用于指定该函数是JNI函数。表示此函数可以被外部调用,在Android开发中不可省略

JNICALL

宏定义,用于指定该函数是JNI函数。,无实际意义,但是不可省略

JNIEnv env

JNIEnv 代表了JNI的环境,只要在本地代码中拿到了JNIEnv和jobject,JNI层实现的方法都是通过JNIEnv 指针调用JNI层的方法访问Java虚拟机,进而操作Java对象,这样就能调用Java代码了。

Java_ + JNI方法所在的完整的类名,把类名里面的”.”替换成”_” + 真实的JNI方法名,这个方法名要和Java代码里面声明的JNI方法名一样+ JNI函数必须的默认参数(JNIEnv* env, jobjectthiz)

env参数是一个指向JNIEnv函数表的指针,

thiz参数代表的就是声明这个JNI方法的Java类的引用

msg参数就是和Java声明的JNI函数的msg参数对于的JNI函数参数

————————————————

开发的基本流程

新版本不用javap生成

1.打印出来的值,是16进制的(日志打印)

原因是:使用的占位符不对%d和%p的区别

LOGD("x=%p", p); // 地址

LOGD("指针取值%d", *p);

2022-10-12 19:59:32.197 16723-16723/com.yuedong.ndkmap D/System.out.c: x=20

2022-10-12 19:59:32.197 16723-16723/com.yuedong.ndkmap D/System.out.c: x=0x7ff7d3906c

2022-10-12 19:59:32.197 16723-16723/com.yuedong.ndkmap D/System.out.c: 指针取值0x14

https://www.cnblogs.com/fang-note/p/14136152.html

#include

#include

#include

#define LOG_TAG "System.out.c"

#define LOGD(...) __android_log_print(ANDROID_LOG_DEBUG, LOG_TAG, __VA_ARGS__)

#define LOGI(...) __android_log_print(ANDROID_LOG_INFO, LOG_TAG, __VA_ARGS__ )

extern "C" JNIEXPORT jstring JNICALL

Java_com_yuedong_ndkmap_MainActivity_stringFromJNI(

JNIEnv *env,

        jobject /* this */) {

std::string hello ="被点击了哈";

    int i =0;

    printf("peng %d\n", i);

    int *p = &i;

    LOGD("x=%d",i);

    LOGD("x=%p",p);

    return env->NewStringUTF(hello.c_str());

}

参考博客:

https://juejin.cn/post/7039948455259111461

--------------------------------------------------------------------------------

3.JNI的数据类型包含两种: 基本类型和引用类型

基本类型主要有jboolean, jchar, jint等

一些基本的数据类型! JString和jint

jint不用转int么?

--------------------------------------------------------------------------------

C函数不允许函数重载, 因为根据签名,

// JNIEXPORT:在Jni编程中所有本地语言实现Jni接口的一个标志// jstring:对应 java 中的数据类型 String// JNICALL:也是一个标记可以去掉,编译运行也不会有问题// JNIEnv:c 与 java 相互调用的桥梁,它提供了很多函数方法// jobj:java 传递下来的对象,即上面的 NdkTest

普通方法用的是jobject . 如果是静态方法是jclass 

举例: 

public native StringgetCMethod();

public static native void changeName();

JNIEXPORT jstring JNICALL

Java_com_yuedong_ndkmap_NdkLibray_getCMethod(JNIEnv *env, jobject thiz) {

// TODO: implement getCMethod()

}

extern "C"

JNIEXPORT void JNICALL

Java_com_yuedong_ndkmap_NdkLibray_changeName(JNIEnv *env, jclass clazz) {

extern "C" JNIEXPORT jstring JNICALL

Java_com_yuedong_ndkmap_MainActivity_stringFromJNI(

JNIEnv *env,

        jobject /* this */) {

std::string hello ="被点击了哈";

注意: JNI数组的处理

https://www.jianshu.com/p/654902148113

https://juejin.cn/post/7040429905191895054

releaseInArrayElements

通过GetIntArrayElements拿到C类型的数组的指针,然后才能进行C数组的处理。

C拿到Java的数组进行操作或者修改以后,需要调用ReleaseIntArrayElements进行更新,这时候Java的数组也会同步更新过来。

这个方法的最后一个参数是模式:

模式作用

0Java数组进行更新,并且释放C/C++数组。

JNI_ABORTJava数组不进行更新,但是释放C/C++数组。

JNI_COMMITJava数组进行更新,不释放C/C++数组(函数执行完,数组还是会释放)。

=================================================================================

知识拓展

C语言和.h文件解析

.h文件详细解析:

/* DO NOT EDIT THIS FILE - it is machine generated */

#include

/* Header for class ndkdemo_peng_cx_com_myapplication_JNIUtils */

#ifndef _Included_ndkdemo_peng_cx_com_myapplication_JNIUtils

#define _Included_ndkdemo_peng_cx_com_myapplication_JNIUtils

#ifdef __cplusplus

extern "C" {

#endif

/*

* Class:    ndkdemo_peng_cx_com_myapplication_JNIUtils

* Method:    getString

* Signature: ()Ljava/lang/String;

*/

JNIEXPORT jstring JNICALL Java_ndkdemo_peng_cx_com_myapplication_JNIUtils_getString

(JNIEnv *, jobject);

#ifdef __cplusplus

}

#endif

#endif

其实 ,我们不用javah命令 , 也能写出头文件 

命名都是有规律 :函数名称规则:Java_完整类名_方法名 , 包名的.号 , 以`_`表示

首先我们需要在Java代码里面声明Native方法原型,比如:

public native void helloJNI(String msg); 

其次我们需要在C/C++代码里面声明JNI方法原型,比如:

extern "C"

JNIEXPORT void JNICALL

Java_com_kgdwbb_jnistudy_MainActivity_helloJNI(JNIEnv* env, jobject thiz,jstring msg) {

    //do something

}

相关文章

网友评论

      本文标题:2. 【干货】火爆全网的《超全NDK精品教程》JNI 基础 (i

      本文链接:https://www.haomeiwen.com/subject/srrqrdtx.html