Android Studio 之 JNI 开发详解

前言

  1. 什么是NDK?
    NDK全称是Native Development Kit,NDK提供了一系列的工具,帮助开发者快速开发C(或C++)的动态库,并能自动将so和java应用一起打包成apk。NDK集成了交叉编译器(交叉编译器需要UNIX或LINUX系统环境),并提供了相应的mk文件隔离CPU、平台、ABI等差异,开发人员只需要简单修改mk文件(指出“哪些文件需要编译”、“编译特性要求”等),就可以创建出so。

  2. 为什么使用NDK?
    1.)代码的保护。由于apk的java层代码很容易被反编译,而C/C++库反汇难度较大。
    2.)可以方便地使用现存的开源库。大部分现存的开源库都是用C/C++代码编写的。
    3.)提高程序的执行效率。将要求高性能的应用逻辑使用C开发,从而提高应用程序的执行效率。
    4.)便于移植。用C/C++写得库可以方便在其他的嵌入式平台上再次使用。

  3. 什么是JNI?
    JNI全称为:Java Native Interface。JNI 是本地编程接口,它使得在 Java 虚拟机内部运行的 Java 代码能够与用其它语言(如 C、C++)编写的代码进行交互。

  4. 为什么使用JNI?
    JNI的目的是使java方法能够调用c实现的一些函数。

  5. 安卓中的so文件是什么?
    Android中用到的so文件是一个c++的函数库。在android的JNI中,要先将相应的C语言打包成so库,然后导入到lib文件夹中供java调用。

本例开发环境如下:
操作系统:Mac
开发环境:Android Studio 2.2 Beta3 + NDK r12 + Gradle 2.14.1

NDK安装

  1. 从Android Studio安装(需翻墙)
    1.)打开AndroidStudio,选择顶部工具条,Tools->Android->SDK Manager

    2.)在弹出来的对话框中选择SDK Tools选项卡

    3.)勾选上图中NDK,点击 Apply,开始安装
    4.)安装完成后,重启Android Studio
  2. AndroidDevTools安装
    1.)打开AndroidDevTools网页,选择导航栏中Android SDK Tools->NDK,选择相应平台的NDK开始下载。

    2.)下载完成后,将NDK解压到某个文件夹下,打开Android Studio,选择File->Project Structure

    在弹出来的对话框中,配置NDK路径,如下所示:

JNI开发

下面我们就一步一步来完成一个示例,从C语言编写的程序中获取字符串,然后在TextView上显示出来。

  1. 新建一个Android Project,命名为 MyApplication


    注意:项目路径中不能有空格!

  2. 项目新建完成后,默认为Android视图,这里为了更清楚的展示,我们切换到Project视图。



    项目结构如下:


  3. 在项目gradle.properties文件中加上以下代码,表示我们要使用NDK进行开发。

android.useDeprecatedNdk=true
  1. 在项目local.properties中加入ndk和sdk的路径:
sdk.dir=/Users/用户名/android-sdk-macosx
ndk.dir=/Users/用户名/android-sdk-macosx/ndk-bundle
  1. 在app文件夹下的build.gradle中的defaultConfig里加入如下代码
ndk{    
      moduleName "hello"       //生成的so文件名字,调用C程序的代码中会用到该名字    
      abiFilters "armeabi", "armeabi-v7a", "x86" //输出指定三种平台下的so库
}

如下所示:


  1. 打开布局文件activity_main.xml,我们来添加一个TextView显示从C程序中返回的字符串
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:id="@+id/activity_main"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:paddingBottom="@dimen/activity_vertical_margin"
    android:paddingLeft="@dimen/activity_horizontal_margin"
    android:paddingRight="@dimen/activity_horizontal_margin"
    android:paddingTop="@dimen/activity_vertical_margin"
    tools:context="me.jockio.myapplication.MainActivity">

    <TextView
        android:id="@+id/textView"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:textSize="20sp" />
</RelativeLayout>
  1. 打开MainActivity.java,添加如下代码:
public class MainActivity extends AppCompatActivity {

    //固定写法,表示我们要加载的资源文件为libhello.so
    static {
        System.loadLibrary("hello");
    }

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        TextView textView = (TextView) findViewById(R.id.textView);
        textView.setText(getStringFromNative());
    }

    //声明一个本地方法,用native关键字修饰
    public native String getStringFromNative();
}
  1. 生成.h头文件
    打开Android Studio底部的Terminal,默认命令行窗口路径已经在当前项目,输入以下命令:
cd app/src/main/java
javah -jni 包名+类名

执行完上面两条命令后,会自动生成.h文件



生成.h文件内容如下:



这里关键部分就是:
JNIEXPORT jstring JNICALL Java_me_jockio_myapplication_MainActivity_getStringFromNative  (JNIEnv *, jobject);
  1. 新建jni文件夹,并拷贝上面生成的.h文件到jni目录
    选择File->New->Folder->JNI Folder


    在弹出的对话框中勾选Change Folder Location,并在下面输入文件夹名,如下图所示:

  2. 在jni目录下,右键新建C文件,文件名任意,输入如下内容:

//引入上面生成的头文件,并实现头文件中声明的方法
#include "me_jockio_myapplication_MainActivity.h"
JNIEXPORT jstring JNICALL Java_me_jockio_myapplication_MainActivity_getStringFromNative
        (JNIEnv *env, jobject obj){
    char *str="String from native C";
    return (*env)->NewStringUTF(env, str);
}

注意观察函数方法名为:Java_包名_类名_方法名,了解到这些后我们以后就可以不生成.h文件,而是直接去写.c文件了。

  1. 选择 Build->Make Project,看app/build/intermediates/ndk/debug/lib目录下是否生成.so文件,如果没有生成,选择 Build->Clean Project,等clean完成后,再Build->Rebuild Project,一般经过上面两步以后都能够解决问题。


  2. 打开模拟器,运行Android程序。这里可以看到已经从libhello.so文件中读取到字符串,并显示在了TextView中。


参考文章

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,686评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,668评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,160评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,736评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,847评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,043评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,129评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,872评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,318评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,645评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,777评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,470评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,126评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,861评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,095评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,589评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,687评论 2 351

推荐阅读更多精彩内容