android jni 调用非静态函数,Android ndk调用非jni标准so方法

背景:

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

一、准备工作

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

c6f987ccdc15

image

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

c6f987ccdc15

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文件

c6f987ccdc15

生成so文件

四、查看依赖so文件信息

接下来我们的程序想要调用so里面的方法。

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

c6f987ccdc15

执行strings指令

可以看到,

c6f987ccdc15

执行结果

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

五、编写jni的文件

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

c6f987ccdc15

生成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);

}

然后运行:

c6f987ccdc15

运行结果

可以看到运行的结果为10,结果正确。

(运行的时候,如果提示

c6f987ccdc15

运行提示

就去build.gradle中设置

android {

...

packagingOptions {

pickFirst 'lib/xxx/libnative-lib.so'

}

}

即可。

)

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

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值