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
}