美文网首页Android技术知识
Android Studio 运行NDK程序时踩过的那些坑

Android Studio 运行NDK程序时踩过的那些坑

作者: android开发经验总结 | 来源:发表于2017-03-17 16:40 被阅读414次

    首先,我把Windows下NDK开编写JNI的流程说一下,在整个流程中着重点出里面的一些坑,以免其他开发者步我后尘。
    1、NDK环境搭建
    这个就不在这里说了,不是重点,大家自行google、百度。
    2、JNI开发
    1)创建Android项目。
    2)查看项目local.properties中加入ndk和sdk的路径是否正确

         ndk.dir=/Users/userName/AndroidStudioProjects/ndk/android-ndk-r13b
          sdk.dir=/Users/userName/Library/Android/sdk
    

    3)配置项目下的gradle.properties文件,表示我们要使用NDK进行开发。

    android.useDeprecatedNdk=true
    

    4)在moudle根目录下的的build.gradle中的defaultConfig标签内部里加入如下代码

    ndk{    
    moduleName "secret"   //生成的so文件名字,跟System.loadLibrary("secret");  中的名字相对应  
    abiFilters "armeabi", "armeabi-v7a", "x86" //输出指定三种平台下的so库,
    // 还可以添加 'x86_64', 'mips', 'mips64'
    }
    

    5)编写jni代码

    package com.david.ndktest;
    -
    public class MainActivity extends AppCompatActivity {
    
     //使用静态代码块,表示我们要加载的资源文件为libsecret.so
     static {
         System.loadLibrary("secret");
     }
    
     @Override
     protected void onCreate(Bundle savedInstanceState) {
         super.onCreate(savedInstanceState);
         setContentView(R.layout.activity_main);
         TextView tv_msg = (TextView) findViewById(R.id.tv_msg);
         tv_msg.setText(stringFromat());
    
     }
     //声明一个本地方法,用native关键字修饰
     public native String stringFromat();
    }
    

    6)生成.h头文件
    直接使用Android Studio 底部的Terminal,默认命令行窗口路径已经在当前项目,进入到app/src/main/java目录,输入以下命令(固定格式:javah -jni 包名+类名),这个一般的博客网站告诉我们的命令

    javah -jni com.david.ndktest.MainActivity
    

    好了,这里存在几个坑,首先第一个坑
    1、Terminal提示 错误:找不到‘com.david.ndktest.MainActivity’类文件
    原因:java没有包含 Android SDK中的 java 文件
    解决办法:
    找到Android sdk目录,找到platforms文件夹,进入“android-20”(或者其他版本也行),然后找到 “android.jar” 文件,将其添加到计算机环境变量的 CLASSPATH 中。重启一下Terminal或者AndroidStudio
    再次执行“javah -jni com.david.ndktest.MainActivity”,好了这里又出了一个小坑
    2、Terminal提示 错误:编码GBK的不可映射字符
    原因:由于JDK是国际版的,在编译的时候,如果我们没有用-encoding参数指定我们的JAVA源程序的编码格式,则javac.exe首先获得我们操作系统默认采用的编码格式。
    解决办法:应该使用-encoding参数指明编码方式,如:javah -jni -encoding UTF-8 com.david.ndktest.MainActivity

    好了,以上两个坑解决掉了,再次执行命令javah -jni -encoding UTF-8 com.david.ndktest.MainActivity 这次Terminal没有任何输出,说明我们已经成功的在java文件夹下生成了com.david.ndktest.MainActivity.h头文件,如下:

    头文件.png
    我们把头文件名字改成secret.c(一定要改成.c文件才能编译), Build ->Build Project 一下工程,但是问题来了,我们并没有在项目的build\intermediates\目录下看到ndk文件夹,说明编译之后没有生成相应的so库,这是第三个坑
    3、原因:secret.c文件在java文件夹下都是java源代码,AndroidStudio无法识别其中的c文件,自然不会去编译。
    解决办法:将命令:javah -jni -encoding UTF-8 com.david.ndktest.MainActivity 改成
    javah -jni -d ../jni -encoding UTF-8 com.david.ndktest.MainActivity 这样就会在java目录的上一级main目录下自动生成jin文件夹,并将生成的头文件放到里面。
    好了,我们在执行一下命令,然后把jni文件夹下生成的文件的文件名改成secret.c,再Build ->Build Project 一下,在build\intermediates\ndk文件夹下看到了打包好的so库,好了,到这里终于松了一口气。
    7)如果6)无法进行,可以直接从其他地方复制头文件进来,或者自己根据格式写一个.c的文件,放到src/main/jni目录下;
    如遇到编译错误,如下:
    图片.png
    解决办法:
    图片.png

    加上红线标记的代码即可,因为jni每次运行都会在build\intermediates\ndk中自动编译生成一个Android.mk文件,这样我们之前在jni中手动编译的Android.mk就没有作用了,加上这句后就不会自动编译了,用的是我们自己的mk文件。

    8)在secret.c文件里实现stringFormat方法如下:

    图片.png

    9)编译C/C++文件,输出 .so文件(第六步无法生成so文件条件下,采用如下方式)
    1、cd命令进入创建的jni文件夹下;
    2、执行ndk-build命令

    若出现如下图所示错误:
    图片.png
    解决办法:

    在build\intermediates\ndk\debug目录下找到Android.mk文件,将其复制黏贴到src\main\jni文件加下,再次执行ndk-build即可自动生成so文件,可在src\mian\libs目录下查看。

    运行该工程,OK,native方法调用成功,效果如下:

    图片.png

    注意:
    1、build.gradle 里面的moduleName "xxx" 跟System.loadLibrary("xxx");的名字要对应起来,但是c文件的名字可以自定义,不一定要一样。
    2、相关参考资料来自
    http://blog.csdn.net/qq_36788768/article/details/59487103
    http://www.2cto.com/kf/201706/648968.html

    相关文章

      网友评论

        本文标题:Android Studio 运行NDK程序时踩过的那些坑

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