栏目分类:
子分类:
返回
名师互学网用户登录
快速导航关闭
当前搜索
当前分类
子分类
实用工具
热门搜索
名师互学网 > IT > 软件开发 > 后端开发 > Java

android NDK 编程学习记录

Java 更新时间: 发布时间: IT归档 最新发布 模块sitemap 名妆网 法律咨询 聚返吧 英语巴士网 伯小乐 网商动力

android NDK 编程学习记录

问题1:c++ string 转 char*

data() 方法:

string str = "hello";
const char* p = str.data();//加const  或者用char * p=(char*)str.data();的形式

c_str()方法:

string str=“world”;
const char *p = str.c_str();//同上,要加const或者等号右边用char*

参考博客

问题2:jstring 转为 char*
const char* GetStringUTFChars(jstring string, jboolean* isCopy)

方法说明:
j_str jstring 类型(Java传递给本地代码的字符串指针)

isCopy: 取值JNI_TRUE和JNI_FALSE,如果值为JNI_TRUE,表示返回JVM内部源字符串的一份拷贝,并为新产生的字符串分配内存空间。如果值为JNI_FALSE,表示返回JVM内部源字符串的指针,意味着可以通过指针修改源字符串的内容,不推荐这么做,因为这样做就打破了Java字符串不能修改的规定。但我们在开发当中,并不关心这个值是多少,通常情况下这个参数填NULL即可。

因为Java默认使用Unicode编码,而C/C++默认使用UTF编码,所以在本地代码中操作字符串的时候,必须使用合适的JNI函数把jstring转换成C风格的字符串。JNI支持字符串在Unicode和UTF-8两种编码之间转换,GetStringUTFChars可以把一个jstring指针(指向JVM内部的Unicode字符序列)转换成一个UTF-8格式的C字符串。在上例中sayHello函数中我们通过GetStringUTFChars正确取得了JVM内部的字符串内容。

问题3:打印Log
  1. 导入头文件:#include
  2. 方法:
std::string hello = "Hello from C++";
__android_log_print(ANDROID_LOG_ERROR, kTAG, "%s", hello.c_str());
  1. 使用宏定义简化方法的使用
#define LOGE(...) ((void)__android_log_print(ANDROID_LOG_ERROR, kTAG, __VA_ARGS__))

std::string hello = "Hello from C++";
LOGE("%s", hello.data());
问题4:meset 函数

1. 函数原型:

# include 
// 虽然参数 c 要求是一个整数,但是整型和字符型是互通的。但是赋值为 '' 和 0 是等价的,
// 因为字符 '' 在内存中就是 0。所以在 memset 中初始化为 0 也具有结束标志符 '' 的作用,
// 所以通常我们就写“0”
void *memset(void *s, int c, unsigned long n);

2. 函数功能:
将s所指向的某一块内存中的前n个字节的内容全部设置为ch指定的ASCII值, 第一个值为指定的内存地址,块的大小由第三个参数指定,这个函数通常为新申请的内存做初始化工作, 其返回值为指向s的指针,它是对较大的结构体或数组进行清零操作的一种最快方法。

问题5:JavaVM 和 JNIEnv

1. JavaVM
JavaVM 是虚拟机在JNI层的代表,一个进程只有一个JavaVM,所有的线程共用一个JavaVM
2. JNIEnv
JNIEnv 表示Java调用native语言环境,是一个封装了几乎全部JNI方法的指针。
JNIEnv只在创建它的线程生效,不能跨进程传递,不同线程的JNIEnv彼此独立。
native环境创建的线程,如果需要访问JNI,必须要调用AttachCurrentThread关联,并使用DetachCurrentThread 解除连接。

JNIEXPORT jint JNICALL
JNI_OnLoad(JavaVM *vm, void *reserved) {
    JNIEnv *env = NULL;
    jint result = -1;
    if (vm->GetEnv((void **) &env, JNI_VERSION_1_6) != JNI_OK) {
        return result;
    }
    
    // 返回jni的版本
    return JNI_VERSION_1_6;
}
问题6 :JNIEnv* # NewGlobalRef {全局引用}
    全局引用 作用域 :
            空间 : 可以 跨方法 , 跨线程使用
            时间 : 创建后可以使用 , 手动释放后全局引用失效

    全局引用创建 : NewGlobalRef
    全局引用释放 : DeleteGlobalRef
    全局引用会阻止 JVM 回收该引用
        //生成局部引用 , 该局部引用使用完毕后可释放
        jclass tmp_class = env->FindClass("kim/hsl/jni/Teacher");

        //将上述生成的局部引用变成 全局引用
        //全局引用释放时 , env->DeleteGlobalRef(class_teacher_global) 即可释放下面转换的 全局引用
        class_teacher_global = static_cast(env->NewGlobalRef(tmp_class));

        //将局部引用释放掉
        env->DeleteLocalRef(tmp_class);

参考博客

问题7:JNI 创建java对象
    jclass cls = env->FindClass("com/example/myapplication/Util");

    //调用无参构造函数
    //构造函数是没有函数名的, 就代表构造函数. ()V 代表无参无返回值
    jmethodID init = env->GetMethodID(cls, "", "()V");
    //调用有参构造函数,两个参数,一个是 String, 一个是 Int
    jmethodID init = env->GetMethodID(cls, "", "(Ljava/lang/String;I)V");
	//创建实例
    jobject util = env->NewObject(cls, init);

参考博客

问题8:JNI 方法签名的类型
对应于Java端的数据类型,我们也可以看一下下面的表: 
Java 类型 类型签名
boolean Z
byte    B
char    C
short   S
int     I
long    L
float   F
double  D
类      L全限定名;,比如String, 其签名为Ljava/lang/util/String;
数组    [类型签名, 比如 [B

参考博客

问题9:JNI 调用 static Java 方法
// 获取 getBuildVersion 方法
// ()Ljava/lang/String; 一定不能少分号
jmethodID versionFun = env->GetStaticMethodID(g_ctx.jniHelperClz, "getBuildVersion", "()Ljava/lang/String;");
问题10:Android native 线程 pthread
  • 创建线程
int pthread_create(pthread_t *thread, const pthread_attr_t *attr, void *(*start_routine)(void*), void *arg); 

函数参数:

  1. 线程句柄 thread:当一个新的线程调用成功之后,就会通过这个参数将线程的句柄返回给调用者,以便对这个线程进行管理。
  2. 入口函数 start_routine(): 当你的程序调用了这个接口之后,就会产生一个线程,而这个线程的入口函数就是start_routine()。如果线程创建成功,这个接口会返回0。
  3. 入口函数参数 *arg : start_routine()函数有一个参数,这个参数就是pthread_create的最后一个参数arg。这种设计可以在线程创建之前就帮它准备好一些专有数据,最典型的用法就是使用C++编程时的this指针。start_routine()有一个返回值,这个返回值可以通过pthread_join()接口获得
  4. 线程属性 attr: pthread_create()接口的第二个参数用于设置线程的属性。这个参数是可选的,当不需要修改线程的默认属性时,给它传递NULL就行。具体线程有那些属性,我们后面再做介绍。
问题11:pthread_create 执行崩溃问题

线程的执行函数需要有返回值

// 错误:
void*  pthreadMethod(void* args) {
    __android_log_print(ANDROID_LOG_ERROR, kTAG, "%s", "updateTicks");
	
}

// 正确:
void*  pthreadMethod(void* args) {
    __android_log_print(ANDROID_LOG_ERROR, kTAG, "%s", "updateTicks");
	
	return nullptr;
}

参考博客

转载请注明:文章转载自 www.mshxw.com
本文地址:https://www.mshxw.com/it/838293.html
我们一直用心在做
关于我们 文章归档 网站地图 联系我们

版权所有 (c)2021-2022 MSHXW.COM

ICP备案号:晋ICP备2021003244-6号