Android NDK:在Android Studio下的基本开发步骤和基础知识点总结

-d 输出目录,jni是gradle默认的路径

-classpath jar的路径,有时碰到的找不到Activity的类的错误可能是由这个引起的

com.zhuanghongji.ndkdemo.JNITest 包名+类名

执行上述命令发现在main目录下多了一个jni文件夹,而且里面有生成好的头文件:com_zhuanghongji_ndkdemo_JNITest.h

内容如下:

/* DO NOT EDIT THIS FILE - it is machine generated */

#include <jni.h>

/* Header for class com_zhuanghongji_ndkdemo_JNITest */

#ifndef _Included_com_zhuanghongji_ndkdemo_JNITest

#define _Included_com_zhuanghongji_ndkdemo_JNITest

#ifdef __cplusplus

extern “C” {

#endif

/*

  • Class: com_zhuanghongji_ndkdemo_JNITest

  • Method: getStringFromJNI

  • Signature: ()Ljava/lang/String;

*/

JNIEXPORT jstring JNICALL Java_com_zhuanghongji_ndkdemo_JNITest_getStringFromJNI

(JNIEnv *, jobject);

#ifdef __cplusplus

}

#endif

#endif

上面代码中的JNIEXPORTJNICALL 是jni的宏,在android的jni中不需要,当然写上去也不会有错。从上面的源码中可以看出这个函数名那是相当的长啊。。。。 不过还是很有规律的, 完全按照:java_pacakege_class_mathod 形式来命名。

注意下上面的注释:

/*

  • 1.该方法所在的类

  • Class: com_zhuanghongji_ndkdemo_JNITest

  • 2.该方法所在类中的对应方法

  • Method: getStringFromJNI

  • 3.其中()表示函数的参数为空

  • 其中Ljava/lang/String;表示函数的返回值是java的String对象

  • Signature: ()Ljava/lang/String;

  • 这里为空是指除了JNIEnv *, jobject 这两个参数之外没有其他参数,

  • JNIEnv*, jobjec
    t是所有jni函数必有的两个参数,

  • 分别表示jni环境和对应的java类(或对象)本身

*/

6.接着在jni目录下新建一个 .c文件

来实现头文件里面声明的方法。

我自己建立的是:com_zhuanghongji_ndkdemo_JNITest.c

实现代码后,如下:

// include刚才生成的头文件

#include “com_zhuanghongji_ndkdemo_JNITest.h”

JNIEXPORT jstring JNICALL Java_com_zhuanghongji_ndkdemo_JNITest_getStringFromJNI

(JNIEnv *env, jobject obj){

return (*env)->NewStringUTF(env, “I’m native function: getStringFromJNI() !”);

}

注意:在jni下面再建一个空的.c文件,如temp.c

要不编译有问题,谷歌官方有提到这个问题,可能是android studio的一个bug。

7.在 local.properties 文件中设置ndk的路径:

我的是:ndk.dir=C:\\Android\\android-ndk-r10

8.在gradle.properties文件进行配置”使用NDK”

此文件末尾增加代码:android.useDeprecatedNdk=true

9.在app目录下的 build.gradle中设置库文件名(生成的so文件名):

工程中共有两个build.gradle配置文件,我们要修改的是在<Project>\app\build.gradle这个文件

找到 defaultConfig 这项,在里面添加如下内容:

ndk{

moduleName “MyJniName” //设置库(so)文件名称,加载时会被用到

ldLibs “log”, “z”, “m” //链接时使用到的库,对应android.mk文件中的LOCAL_LDLIBS

abiFilters “armeabi”, “armeabi-v7a”, “x86” //最终输出指定三种abi体系结构下的so库,目前可有可无

}

这时,再执行”Build->Make Project”,就可以编译出so文件了。

编译出来的库文件被Studio输出到了下图的路径中:

这里写图片描述

10.在JNITest.java中增加对so的加载:

static {

System.loadLibrary(“MyJniName”);

}

至此:JNITest.java 的完整代码如下:

// JNITest.java

package com.zhuanghongji.ndkdemo;

public class JNITest {

static {

System.loadLibrary(“MyJniName”);

}

public native String getStringFromJNI();

}

11.现在在MainActivity中使用JNITest类的native方法:

package com.zhuanghongji.ndkdemo;

import android.os.Bundle;

import android.support.v7.app.AppCompatActivity;

import android.widget.TextView;

public class MainActivity extends AppCompatActivity {

private TextView mTextView;

@Override

protected void onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

setContentView(R.layout.activity_main);

mTextView= (TextView) findViewById(R.id.tv);

String s = new JNITest().getStringFromJNI();

mTextView.setText(s);

}

}

现在运行的话,你就会看到下面的结果:

这里写图片描述

上面的字符串I'm native function: getStringFromJNI() !是C文件中的

return (*env)->NewStringUTF(env, “I’m native function: getStringFromJNI() !”);

这段代码返回的。

三、基本知识点总结


1.为什么使用NDK

  1. 代码的保护。由于apk的java层代码很容易被反编译,而C/C++库反汇难度较大。

  2. 可以方便地使用现存的开源库。大部分现存的开源库都是用C/C++代码编写的。

  3. 提高程序的执行效率。将要求高性能的应用逻辑使用C开发,从而提高应用程序的执行效率。

  4. 便于移植。用C/C++写得库可以方便在其他的嵌入式平台上再次使用。

2.NDK简介

  1. NDK是一系列工具的集合
  • NDK提供了一系列的工具,帮助开发者快速开发C(或C++)的动态库。NDK集成了交叉编译器,并提供了相应的mk文件隔离CPU、平台、ABI等差异,开发人员只需要简单修改mk文件(指出“哪些文件需要编译”、“编译特性要求”等),就可以创建出so。

  • NDK可以自动地将so和Java应用一起打包,极大地减轻了开发人员的打包工作。

  1. NDK提供了一份稳定、功能有限的API头文件声明

Google明确声明该API是稳定的,在后续所有版本中都稳定支持当前发布的API。从该版本的NDK中看出,这些API支持的功能非常有限,包含有:C标准库(libc)标准数学库(libm)压缩库(libz)Log库(liblog)

  • NDK提供了一系列的工具,帮助开发者快速开发C(或C++)的动态库。NDK集成了交叉编译器,并提供了相应的mk文件隔离CPU、平台、ABI等差异,开发人员只需要简单修改mk文件(指出“哪些文件需要编译”、“编译特性要求”等),就可以创建出so。

  • NDK可以自动地将so和Java应用一起打包,极大地减轻了开发人员的打包工作。

  1. NDK提供了一份稳定、功能有限的API头文件声明

Google明确声明该API是稳定的,在后续所有版本中都稳定支持当前发布的API。从该版本的NDK中看出,这些API支持的功能非常有限,包含有:C标准库(libc)标准数学库(libm)压缩库(libz)Log库(liblog)

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值