Android studio,第一个生成,调用成功的jni(说多了都是泪)

0x01 序言:

  泪从何处说起呢?其实很早以前就用过android studio写过c++,但是,但是一直没有成功生成过so文件,所以心中一直有一个纠结。。。为什么不成功呢。。。

直到今天,由于工作的缘故不得不重新拾起,应该是昨天,昨天就在写了,不过,没成功。

0x02 网上的一般性操作

  1、创建一个项目。

  包名姑且用:com.tangh.test_so2

  2、新建一个类,和jni方法。

public class JniUtil {
    static {
        System.loadLibrary("hello");
    }

    public native String getString();
}

  生成(build)一下,查看 项目名称\app\build\intermediates\classes\debug\com\tangh\test_so2\JniUtil.class文件

  3、返回到 classes\debug\下 shift+鼠标右键,在此处打开命令窗口

  4、执行命令:javah -jni com.tangh.test_so2.JniUtil  会在 debug下生成一个  com_tangh_test_so2_JniUtil.h文件。。。

  

extern "C" {
#endif
/*
 * Class:     com_tangh_test_so2_JniUtil
 * Method:    getString
 * Signature: ()Ljava/lang/String;
 */
JNIEXPORT jstring JNICALL Java_com_tangh_test_1so2_JniUtil_getString
  (JNIEnv *, jobject);

#ifdef __cplusplus
}

  方法名变了。在下划线前面加了一个数字,参考:https://blog.csdn.net/sambillyr/article/details/48864189

  5、然后写.cpp文件。实现它

#include <jni.h>
#include "com_tangh_test_so2_JniUtil.h"
JNIEXPORT jstring JNICALL Java_com_tangh_test_1so2_JniUtil_getString
  (JNIEnv *env, jobject thiz){
     return env->NewStringUTF("hello jni!");
  }

  处于,好看,我把函数名字改动了一下,去掉了那个1,这就是引发了后续的一个问题了。。。。。

  6、在项目/src/app/main/下新建一个文件夹 jni。然后把.h文件.cpp文件,都存放进入,听过好像jni本身有bug,需要一个空的.c文件,于是我写了一个空的.cpp,空的.c文件

empty.c 和 FixBug.cpp
 7、当然,我是一直没成功,于是手动创建了。Android.mk和一个Application.mk文件
Android.mk
LOCAL_PATH := $(call my-dir)

include $(CLEAR_VARS)

LOCAL_MODULE := hello
LOCAL_SRC_FILES := com_tangh_test_so2_JniUtil.cpp  \
empty.c  \
FixBug.cpp

include $(BUILD_SHARED_LIBRARY)

  Application.mk

APP_PLATFORM = android-16
APP_ABI := armeabi-v7a

  8、顺便改一下。build.gradle...

defaultConfig {
        applicationId "com.tangh.test_so2"
        minSdkVersion 16
        targetSdkVersion 24
        versionCode 1
        versionName "1.0"
        ndk{
            moduleName "hello"
            abiFilters "armeabi-v7a", "x86"
        }
    }
    buildTypes {
        release {
            minifyEnabled false
            proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
        }
    }
    sourceSets.main {
        jniLibs.srcDir '/src/main/jniLibs'
        jni.srcDirs = [] //disable automatic ndk-build call
    }

// 部分内容省略了。

  。。。jni.srcDirs设置成空,意思就是手动生成so.

  给路径就是告诉ide,你来给我生成。当然,两种方法都试了。都有错误。那么不生成。要么我在jni目录下执行ndk-build命令。。报错 XXXXXXXXXXX Error 1..

我一直以为是nkd的错误与,或者是我哪里配置没有弄好。。。。

  9、Q群求助。。结果别人居然可以编译,我就纳闷了,于是把ndk拷贝到,虚拟机,安装jdk,真的可以。。。。原谅我的PC机,他已经老了,可能年久失修,,,

  10,好吧,算解决了。但是 so生成了,拷贝回来了。放在了。目录/src/main/jniLibs/armeabi-v7a/libhello.so ....但是,编译之后没问题。

运行起来,app退出了,找不到native函数,我用ida看了一下,有这个函数啊,,,,想到,函数名称,把“.”,变成了下划线"_"...难道是我的下划线。。。。

突然心中一万只(XXX)飞过,好吧,百度一下。改回去,重新生成,重新。。。,,,成功了。。

  11.。真的成功了,突然成就感爆棚。。。。好吧,这个问题,我困扰我很久了。发个红包,庆祝一下。

 

0x02 总结

  有时候,一个问题,不一定是代码的问题,环境也可能是一个困扰你不得其所的大问题。换个思路,让别人也试试,说不定你会有新发现。同时也告诉我们,作为一个程序员,应该有多个环境。

一个干净的环境,能够更好地甄别问题所在,而不是像我的PC机一样,1T的硬盘,只有不到100G了,大多数软件都装了。。这么说吧。

vs2010,2012,2013,2015都装了。。。。

 

posted @ 2018-06-15 17:10  Supper_litt  阅读(591)  评论(0编辑  收藏  举报