Android Studio JNI使用之个人笔记

来源:互联网 发布:游戏编程需要数学吗 编辑:程序博客网 时间:2024/05/24 01:00

JNI数据类型

上面我们提到JNI定义了一些自己的数据类型。这些数据类型是衔接Java层和C/C++层的,如果有一个对象传递下来,那么对于C/C++来说是没办法识别这个对象的,同样的如果C/C++的指针对于Java层来说它也是没办法识别的,那么就需要JNI进行匹配,所以需要定义一些自己的数据类型。

1.原始数据类型

Java TypeNative TypDescriptionbooleanjbooleanunsigned 8 bitsbytejbytesigned 8 bitscharjcharunsigned 16 bitsshortjshortsigned 16 bitsintjintsigned 32 bitslongjlongsigned 64 bitsfloatjfloat32 bitsdoublejdouble64 bitsvoidvoidN/A

2.引用类型

 前面我们在获取AndroidJni对象的使用通过定义jclass引用,然后调用FindClass函数获取了该对象,所以JNI也定义了一些引用类型以便JNI层调用,具体的引用类型如下:

jobject                     (all Java objects)||-- jclass                    (java.lang.Class objects)|-- jstring                    (java.lang.String objects)|-- jarray                    (array)|      |--jobjectArray       (object arrays)|      |--jbooleanArray        (boolean arrays)|      |--jbyteArray            (byte arrays)|      |--jcharArray            (char arrays)|      |--jshortArray        (short arrays)|      |--jintArray            (int arrays)|     |--jlongArray            (long arrays)|      |--jfloatArray        (float arrays)|      |--jdoubleArray         (double arrays)||--jthrowable

3.方法和变量的ID
 当需要调用Java中的某个方法的时候我们首先要获取它的ID,根据ID调用JNI函数获取该方法,变量的获取过程也是同样的过程,这些ID的结构体定义如下:

struct _jfieldID;              /* opaque structure */ typedef struct _jfieldID *jfieldID;   /* field IDs */ struct _jmethodID;              /* opaque structure */ typedef struct _jmethodID *jmethodID; /* method IDs */

Android Stduio中JNI应用

新建JniUtils类实现native方法

public class JniUtils {    public static native String getStringFormC();}

然后clean project 再rebuild project 生成class文件,


再打开Terminal输入指令
cd app\src\main
然后再输入指令
javah -d jni -classpath ..\..\build\intermediates\classes\debugcom.dongdong.ndkjnidemo.JniUtils

这个时候在src/main/jni下就会生成.h文件,新建一个c类随便取一个名字,添加代码如下

#include "stdio.h" 
#include "stdlib.h" 
#include "com_dongdong_ndkjnidemo_JniUtils.h" 
/* * Class: Java_com_dongdong_ndkjnidemo_JniUtils * Method: getStringFormC * Signature: ()Ljava/lang/String; */JNIEXPORT jstring JNICALL Java_com_dongdong_ndkjnidemo_JniUtils_getStringFormC (JNIEnv *env, jobject obj){ return (*env)->NewStringUTF(env,"这里是来自c的string");

这里发现头文件#include <jni.h>报红色,是因为咱们还没有配置ndk环境,打开file->project structure选择你所下载的ndk环境路径

设置好了之后,发现头文件还是红色的,然后再build一下工程,就会有提示

Error: NDK integration is deprecated in the current plugin. Consider trying the new experimental plugin. For details, see http://tools.android.com/tech-docs/new-build-system/gradle-experimental. Set "android.useDeprecatedNdk=true" in gradle.properties to continue using the current NDK integration.

按着提示做就行了 在gradle.properties文件末尾添加android.useDeprecatedNdk=true就ok啦
然后在app文件下得build.gradle ->defaultConfig括号内添加如下代码

ndk {            moduleName "Lib"            ldLibs "log"            abiFilters "armeabi", "armeabi-v7a"       }//输出指定三种abi体系结构下的so库,目前可有可无。

到了这一步重新build项目,发现已经没有变红了。接下来就是运用了,在JniUtils类里面添加如下代码

static {       System.loadLibrary("NdkJniDemo");//之前在build.gradle里面设置的so名字,必须一致}

然后简单调用就行了,MainActivity代码如下

public class MainActivity extends AppCompatActivity {    TextView textView;    @Override    protected void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        setContentView(R.layout.activity_main);        textView = (TextView) findViewById(R.id.ndk_text);        textView.setText(JniUtils.getStringFormC());    }}

咱们打开app->intermediates-ndk-debug发现生成了三个文件夹,并且对应了之前我们在build.gradle配置的abiFilters
大功告成。

Eclipse下需要cd 项目目录 javah -d jni -classpath bin/classes com.dongdong.utils.MainActvity生成.h文件。创建.c文件实现相关代码。修改.mk文件。其中
LOCAL_MODULE     := ApkPatchLibrary     //库名称
Android.mk文件LOCAL_SRC_FILES  := com_dongdong_utils_MainActvity.c  C文件
LOCAL_LDLIBS     := -lz -llog  //Android 日志配置

Application.mkAPP_ABI:=armeabi armeabi-v7a arm64-v8a  //编译的平台

最后cd 项目目录 执行ndk-build命令即可。


0 0
原创粉丝点击