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 <jni.h>
#include <string>
#include <test.h>//導入需要的.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庫和其中的方法,復雜的原理基本一致。

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。

推薦閱讀更多精彩內容