Android ndk调用非jni标准so方法

背景:

最近遇到了一个问题,需要调用第三方的so方法,但这个so内的方法不是标准的jni方法。这就需要我们自己写jni然后,链接到已经存在的so库,通过jni调用so库中的方法,就可以实现我们的需求。

一、准备工作

1.工作环境,当前是mac10.13+AndroidStudio3.1(开始使用的3.4版本,但遇到了些问题)

image

一路next之后,完成项目的创建。

image

二、编写依赖的c++文件(非标准jni方式)

test.h文件


#ifndef _TEST_JNI_ADD_H_
#define _TEST_JNI_ADD_H_

class Add {

public:
    Add();

    ~Add();

    int add(int x, int y);

};
#endif

test.cpp文件

#include "test.h"

Add::Add() {
}
Add::~Add() {
}


int Add::add(int x, int y) {
    return x + y;
}

三、编写依赖的cmake文件

#指定cmake最小版本
cmake_minimum_required(VERSION 3.4.1)

#设置生成的so动态库最后输出的路径
set(CMAKE_LIBRARY_OUTPUT_DIRECTORY ${CMAKE_SOURCE_DIR}/src/main/jniLibs/${ANDROID_ABI})

#生成so
add_library( # 设置生成库的名字
             add

             # 生成动态库
             SHARED

             # 指定源码文件,这里指定test.cpp文件
             src/main/cpp/test.cpp )

#依赖的头文件
INCLUDE_DIRECTORIES(${CMAKE_CURRENT_SOURCE_DIR}/src/main/cpp)


find_library( # log库的别名
              log-lib

             #log库
              log )


#链接代码到指定的库
target_link_libraries( # Specifies the target library.
                       add

                       # Links the target library to the log library
                       # included in the NDK.
                       ${log-lib} )

点击build->Make Project,可以看到在jniLibs文件夹下生成了so文件


生成so文件

四、查看依赖so文件信息

接下来我们的程序想要调用so里面的方法。
有时候我们并不知道第三方so中的方法名是什么,有什么参数,可以在命令行中使用strings或者nm指令查看。我们进到so的目录下执行strings libadd.so。


执行strings指令

可以看到,


执行结果

so中的类库依赖关系和类名方法名都可以看到,这个就是我们接下来要依赖的送文件。

五、编写jni的文件

我们想要访问so的方法,就需要通过android studio,生成native方法:


生成native方法

下面编写jni,这里直接调用了so中的Add类的add方法

#include 
#include 
#include //导入需要的.h文件,这个是必须的,如果依赖的第三方库没有.h,需要自己编写


extern "C"
JNIEXPORT jint JNICALL
Java_com_example_jnitest_jnidemo_MainActivity_nativeAdd(JNIEnv *env, jobject instance, jint a,
                                                        jint b) {

    //生成add对象并调用方法
    Add addObj;
    int result = addObj.add(a,b);

    return result;

    
}

五、编写jni的cmake文件

#指定cmake最小版本
cmake_minimum_required(VERSION 3.4.1)

#设置生成的so动态库最后输出的路径
set(CMAKE_LIBRARY_OUTPUT_DIRECTORY ${CMAKE_SOURCE_DIR}/src/main/jniLibs/${ANDROID_ABI})


#指定文件夹下所有的c和c++文件
#file(GLOB_RECURSE cpp_srcs "src/main/cpp/*.c" "src/main/cpp/*.cpp")

#message会生成日志,放在.extrrnalNativeBuild/cmake/debug/xxxx/cmake_build_output.txt文件中
#message(STATUS "src_files ${cpp_srcs}")


#生成so
add_library( # 设置生成库的名字
             native-lib

             # 生成动态库
             SHARED

             # 指定源码文件,这里指定test.cpp文件
             src/main/cpp/native-lib.cpp )

#依赖的头文件
INCLUDE_DIRECTORIES(${CMAKE_CURRENT_SOURCE_DIR}/src/main/cpp)


#依赖的add库
add_library(add SHARED IMPORTED)
set_target_properties(add
            PROPERTIES IMPORTED_LOCATION
            ${CMAKE_CURRENT_SOURCE_DIR}/src/main/jniLibs/${ANDROID_ABI}/libadd.so)




find_library( # log库的别名
              log-lib

             #log库
              log )


#链接代码到指定的库
target_link_libraries( # Specifies the target library.
                       native-lib

                       #add库需要链接
                       add

                       # Links the target library to the log library
                       # included in the NDK.
                       ${log-lib} )

完成之后,Build->Make Project,可以看到在jniLibs目录下生成了新的libnative-lib.so文件。

五、调用生成的so文件

在第三部生成jni方法的时候已经通过Sydstem.loadLibrary加载so文件,

package com.example.jnitest.jnidemo;

import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.widget.TextView;

public class MainActivity extends AppCompatActivity {

    // Used to load the 'native-lib' library on application startup.
    static {
        System.loadLibrary("native-lib");//加载so
    }

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        // Example of a call to a native method
        TextView tv = (TextView) findViewById(R.id.sample_text);
        tv.setText(String.valueOf(nativeAdd(4,6)));
    }

    /**
     * A native method that is implemented by the 'native-lib' native library,
     * which is packaged with this application.
     */
    public native int nativeAdd(int a, int b);
}

然后运行:


运行结果

可以看到运行的结果为10,结果正确。
(运行的时候,如果提示


运行提示

就去build.gradle中设置

android {
      ...
         packagingOptions {
        pickFirst 'lib/xxx/libnative-lib.so'
        }
    }

即可。

通过以上的步骤,我们实现了简单的调用第三方so库和其中的方法,复杂的原理基本一致。

你可能感兴趣的:(Android ndk调用非jni标准so方法)