正在准备中的项目里,有一部分打算直接移殖Linux开发组在之前就完成的功能,他们是使用C语言开发。考虑到维护的问题,准备让他们将代码打包成so文件,再引用到我的项目中。这样也就相当于我去引用一个第三方库,并且这个库中的代码格式也不一定是我们JNI开发时规定的命名,因此,需要通过我自己的C文件再去调用so库中的方法。
1 生成SO库
1.1 Native方法
新建项目JNISODemo,在MainActivity中定义Native方法:
public native String getString();
1.2 头文件生成
.h文件的生成是在命令行cd到main目录下,再使用javah生成。
这次是想介绍下快捷方式。
File -> Settings -> Tools -> External tools -> +
Program: $JDKPath$\bin\javah.exe
Arguments: -classpath . -jni -o $ModuleFileDir$\src\main\jni\$Prompt$ $FileClass$
Working directory: $ModuleFileDir$\src\main\java
在我们声明native方法的类上点击右键,javah,输入命名(我命名为Test.h),之后就会先自动创建一个jni文件夹,然后生成一个Test.h文件,copy Test.h,并将命名改为Test.c。
1.3 完成C代码
在Test.c中简单完成下我们定义的方法,返回一个字符串:
#include <jni.h>
/* Header for class com_david_jnisodemo_MainActivity */
#ifndef _Included_com_david_jnisodemo_MainActivity
#define _Included_com_david_jnisodemo_MainActivity
#ifdef __cplusplus
extern "C" {
#endif
/*
* Class: com_david_jnisodemo_MainActivity
* Method: getString
* Signature: ()Ljava/lang/String;
*/
JNIEXPORT jstring JNICALL Java_com_david_jnisodemo_MainActivity_getString
(JNIEnv *env, jobject instance) {
return (*env)->NewStringUTF(env, "This is a test!");
}
#ifdef __cplusplus
}
#endif
#endif
1.4 CMakeList.txt
cmake_minimum_required(VERSION 3.4.1)
add_library( # Sets the name of the library.
Test
# Sets the library as a shared library.
SHARED
# Provides a relative path to your source file(s).
src/main/jni/Test.c )
find_library(log-lib log )
target_link_libraries( # Specifies the target library.
Test
${log-lib} )
1.5 build.gradle
ndk {
ldLibs "log"//实现__android_log_print
abiFilters "armeabi-v7a" //平台配置,因为在Android上,就只写了一个
}
1.6 编译生成so
编译完成后就如下图,产生一个libTest.so的文件,这就是我们要的。把它当做Linux最后打包成的so文件。
2 导入第三方so
新建一个项目JNIUseSoDemo,项目结构如下。同样是在MainActivity中定义Native方法,生成UseSo.c。将我们上一步生成的so文件拷贝到jniLibs下(armeabi-v7a是平台)。以及上一步中的头文件也拷贝到jni下。
2.1 完成C代码
我在UseSo中getString方法去调用了so库中的getString方法。
#include <jni.h>
#include "Test.h" //so库的头文件,必须要引用!
/* Header for class com_david_jniusesodemo_MainActivity */
#ifndef _Included_com_david_jniusesodemo_MainActivity
#define _Included_com_david_jniusesodemo_MainActivity
#ifdef __cplusplus
extern "C" {
#endif
/*
* Class: com_david_jniusesodemo_MainActivity
* Method: getString
* Signature: ()Ljava/lang/String;
*/
JNIEXPORT jstring JNICALL Java_com_david_jniusesodemo_MainActivity_getString
(JNIEnv *env, jobject instance) {
return Java_com_david_jnisodemo_MainActivity_getString(env, instance);
}
#ifdef __cplusplus
}
#endif
#endif
当然还没有完成。
2.2 CMakeList.txt
在CMake中将LibTest.so导入工程
cmake_minimum_required(VERSION 3.4.1)
add_library( # Sets the name of the library.
UseSo
# Sets the library as a shared library.
SHARED
# Provides a relative path to your source file(s).
src/main/jni/UseSo.c )
#导入第三方so包,并声明为 IMPORTED 属性,指明只是想把 so 导入到项目中
add_library( Test
SHARED
IMPORTED )
#指明 so 库的路径,CMAKE_SOURCE_DIR 表示 CMakeLists.txt 的路径
set_target_properties(
Test
PROPERTIES IMPORTED_LOCATION
${CMAKE_SOURCE_DIR}/src/main/jniLibs/armeabi-v7a/libTest.so )
#指明头文件路径,不然会提示找不到 so 的方法
include_directories(src/main/jni/)
find_library(log-lib
log )
target_link_libraries( # Specifies the target library.
UseSo
Test
${log-lib} )
2.3 build.gradle
ndk {
abiFilters 'armeabi-v7a'
}
2.4 最终调用
public class MainActivity extends Activity {
static {
System.loadLibrary("UseSo"); //加载SO库
}
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
Log.e("TEST",getString());
}
public native String getString(); //它会调用Java_com_david_jniusesodemo_MainActivity_getString方法,然后该方法又回去调用so库中的Java_com_david_jnisodemo_MainActivity_getString方法,得到返回字符串。
}
验证没有问题,导入第三方so库完成。
加载so库代码:
static {
loadNativeLibrary();
}
private static synchronized void loadNativeLibrary() {
System.loadLibrary("encrypt");
}
调用的so内的方法前需要添加native关键字:
private static native byte[] e(byte[] input);
需要注意的点:一、包名要和so库中的包名一样;二、类名也要一致。
我们看看so库的头文件中的函数:
JNIEXPORT jint JNICALL Java_cn_scnu_MainActivity_num
(JNIEnv *, jobject);
函数名为Java_包名类名函数名。所以我们创建的Android工程的包名也要一致,即cn_scnu,同时,负责加载并且提供native方法的类的类名也要相同,即MainActivity,否则调用不成功,提示找不到方法。
过程中遇到两个错误:
1.is 32-bit instead of 64-bit动态库错误
2.UnsatisfiedLinkError
有价值的解决方式:
https://segmentfault.com/a/1190000006210166
https://blog.csdn.net/xiaxiayige/article/details/68925669
————————————————
java.lang.UnsatisfiedLinkError: No implementation found for
错误的可能原因:
1、.so 文件路径问题
(1) 要么你就在main/下新建一个JniLibs文件夹,再把SDK里带的so文件目录诸如armeabi等等文件夹丢进去就可以了。
(2) 要不你就把放so文件的目录诸如armeabi直接放在放jar文件的目录libs里头,然后修改build.gradle文件的内容,添加上图或者如下
android { sourceSets{ main{ jniLibs.srcDirs=['Libs'] } }
2、jni 调用类的路径一定要一致。也就是说 .so中函数声明涉及到的package name和class name与调用它的package name和class name不符。因此我们要改变我们工程中的package name和class name。使其与.so文件中函数签名提示的一致,在这个类中加入native方法的声明。
如: JNI接口 Java_com_netease_xtc_cloudmusic_utils_NeteaseMusicUtils_nativeInit 中,com.netease.xtc.cloudmusic.utils 代表的是 package name ,NeteaseMusicUtils 则是 class name。
而第三方提供so库的工作人员的c文件的定义JNI接口为Java_com_netease_cloudmusic_utils_NeteaseMusicUtils_nativeInit,即: package name 必须为: com.netease.cloudmusic.utils ,而class name 必须为 NeteaseMusicUtils 。
3、未被成功调用
public class Native { static { System.loadLibrary("xxxx"); } } 改为: public class Native { Native { System.loadLibrary("xxxx"); } }
版权声明:本文为CSDN博主「buder得儿得儿以得儿以得儿得儿」的原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/cpcpcp123/article/details/87697569