用AS编写第一个so

江南小虫虫 2020年02月13日 43次浏览

我的gradle是3.5.1, Android Studio的版本是3.5.1

找了两个小时的博客, 硬是没找到能用的... 算了, 自力更生, 丰衣足食

0x00 环境配置

  1. 确定安装了ndk, cmake, LLDB

  2. 新建项目的时候, 记得选择Native C++

    之前版本的include c++ support选项在这个版本中已经独立出去了...

  3. 先配置好sdk的路径: File->Project Structure...->SDK Location-> Android SDK Location

    在Android视图下, 打开local.properties, 看到ndk的路径已经配置完成

0x02 声明native方法

新建一个类, 声明native方法. 这个类是java与C/C交互的中介, 方法**由java声明, 由C/C实现**

这里为了方便接下来的操作, 切换成Project视图

public class myJNI {
    static {
        System.loadLibrary("JniTest"); // 要加载的so的名字
    }
    public static native String sayHello();
}

虽然这里的sayHello()方法报错了, 但是不用管

0x03 编译与头文件的生成

  1. 使用javac编译上述文件, 生成class文件

  2. 确认自己的包名和类名, 然后在java目录使用 javah 包名.类名 命令生成.h头文件, 然后就能看到生成了一个.h文件

    注意一定要在java层目录下输入命令, 不然不会报错: 找不到xxx类

  3. 将生成的com_example_test_myJNI.h拖到cpp目录下

  4. 然后删掉原来cpp目录下的native-lib.cpp

0x04 实现头文件的函数

  1. cpp目录下新建main.c

  2. main.c中的内容, 首先是将头文件包括进来, 然后实现头文件中的sayHello方法

    #include "com_example_test_myJNI.h"
    JNIEXPORT jstring JNICALL Java_com_example_test_myJNI_sayHello(JNIEnv *env, jclass jobj)
    {
        return (*env)->NewStringUTF(env,"hello 52pojie!");
    }
    

    注意: 这时候可以看到上面报错了, 先别管, 下面我们就来解决它

  3. 由于我们使用CMake来生成so的, 所以要修改CMakeLists.txt来指定so名称so的源文件的相对路径

    然后sync一下

    发现main.c已经不报错了

0x05 生成so库

Build->Rebuild Project来生成so库

生成的so在app\build\intermediates\cmake\debug\obj\或者app\build\intermediates\merged_native_libs\debug\out\lib或者app\build\intermediates\stripped_native_libs\debug\out\lib\

ps: 这几个路径中的so库, 我暂时不知道区别在哪...哪位大佬知道的话, 麻烦留言说一声.

注意: 这里的so名字 = lib+ 我们在CmakeLists.txt中起的名字 + .so

0x06 配置so库

app/src/main下新建jniLIB目录, 并将生成的SO文件拷贝到该文件夹下

这里的so库, 我测试的时候选择的是app\build\intermediates\merged_native_libs\debug\out\lib路径下的, 其他的请自测

0x07 调用

打开MainActivity插入一条log来调用so中的sayHello方法,并连接手机调试

0x08 传参调用

声明的时候

public class myJNI {
    static {
        System.loadLibrary("JniTest");
    }
    public static native String sayHello();
    public static native int myAdd(int a,int b);
}

定义的时候

#include "com_example_test_myJNI.h"
JNIEXPORT jstring JNICALL Java_com_example_test_myJNI_sayHello(JNIEnv *env, jclass jobj)
{
    return (*env)->NewStringUTF(env,"hello 52pojie!");
}

JNIEXPORT jint JNICALL Java_com_example_test_myJNI_myAdd(JNIEnv *env, jclass jobj, jint a, jint b)
{
    return a+b;
}

调用的时候

public class MainActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        Log.i("开始调用so中的sayhello方法",myJNI.sayHello());
        Log.i("开始调用so中的myAdd方法:",String.valueOf(myJNI.myAdd(1,2)));
    }
}