[Android]JNI疑难问题

1.Window环境下As无法启动jni调试的问题

错误提示
Attention! No symbol directories found - please check your native
根本原因是因为so库被打包为release版本,代码已经无法跟踪到,所以as才提示错误。
解决方案
1.引用so的module的build.gradle当中在android节点中加入publishNonDefault true
2.app.iml的native-android-gradle节点下,查看


        

如果value为release,请改为debug,这里有可能出现无法编译的问题,需要重启As才能正常运行

2.kotlin中Long和java的Long类型到C++中的转换

这里面 kotlin中的Long和java的long类型,到jni是对应jlong类型,但是在C++中是对应long long类型,如果但是使用long类型,会出现溢出问题,这里面请一定要谨慎(查了两天才发现这个问题)
又或者改为int64_t的类型,这样也是相当于long long类型。

3.C++ Native层中读取到转换本地时间

使用jni来读取

JNIEXPORT jlong JNICALL Java_jni_JniTest_getTime
(JNIEnv * env, jclass obj){
    jclass cls = env->FindClass("java/util/Date");
    jmethodID conmid = env->GetMethodID(cls,"","()V");
    jobject obj1 = env->NewObject(cls,conmid);
    jmethodID getTimeID = env->GetMethodID(cls,"getTime","()J");
    jlong time = env->CallNonvirtualLongMethod(obj1,cls,getTimeID);
    return time;
}

gettimeofday()来获取时间

#include 
long long currentTimeInMilliseconds()
{
    struct timeval tv;
    gettimeofday(&tv, NULL);
    return ((tv.tv_sec * 1000) + (tv.tv_usec / 1000));
}

使用C++来读取,这里使用了CLOCK_MONOTONIC来获取时间,要比
gettimeofday()和currentTimeMillis要准确

//获取当前时间,以纳秒为单位
int64_t getTimeNsec() {
    struct timespec now;
    clock_gettime(CLOCK_MONOTONIC, &now);
    return (int64_t) now.tv_sec*1000000000LL + now.tv_nsec;
}

//获取当前时间,以毫秒为单位
int64_t getTimeMise(){
    struct timespec now;
    clock_gettime(CLOCK_MONOTONIC,&now);
    return (int64_t) now.tv_sec *1000LL + now.tv_nsec/1000000;
}

4.C++ Native层打印int64_t的值

#include 
int64_t t;
printf("%" PRId64 "\n", t);

***********2019.5.23***************

5.华为荣耀jni构造方法崩溃

下面的创建方法,实现jni层反射java层的无参构造,结果有可能出现崩溃了。

    jobject newObject(jclass clz) {
        jmethodID init = m_env->GetMethodID(clz, "", "()V");
        return m_env->NewObject(clz, init);
    };

因为如果不写类的无参构造,荣耀无法反射到java类中默认的无参构造方法,所以一定要确保类中主动编写了无参构造方法。
但对于一些java内置的类,例如Integer,默认没有写无参构造的,只能反射其他的构造方法了。

对应荣耀这么广的机型,所以一定要注意。

你可能感兴趣的:([Android]JNI疑难问题)