Android Studio的JNI开发快餐教程

简介: 生命不息,折腾不止,让我们一起跟随Android Studio继续折腾。

Android Studio的JNI开发快餐教程

从eclipse换到Android Studio之后,原来的NDK集成已经不能用了。同时,AndroidStudio也是在快速迭代进步中,不仅支持内置的ndk支持插件,还有支持外部编译系统的plugin. 而且这还是在实验中的结果,在不久的将来,还可能有更新的变化。不过,万变不离其宗,我们打好基础,形式上的东西跟着Android Studio变就好。

旧式的NDK支持

首先说明,既然是Android Studio认为它已经过时了,所以我们需要在gradle.properties中声明一下我们还坚持用这个属性:

android.useDeprecatedNdk=true

下面我们看下,在build.gradle中如何使用这种老式的办法:

apply plugin: 'com.android.application'

    android {
        compileSdkVersion 24
        buildToolsVersion "24.0.0"

        defaultConfig {
            applicationId "com.yunos.xulun.testcppjni"
            minSdkVersion 21
            targetSdkVersion 24
            versionCode 1
            versionName "1.0"
            ndk{
                moduleName "testcppjni"
            }
        }
        buildTypes {
            release {
                minifyEnabled false
                proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
            }
        }
    }

这其中使用的plugin,是"com.android.application".

我们现在可以开始写一个类测试一下:

package com.yunos.xulun.testcppjni;

public class TestCppJni {
    static{
        System.loadLibrary("testcppjni");
    }
    public static native int callCpp();
}

编译之后,使用javah工具生成头文件com_yunos_xulun_testcppjni_TestCppJni.h:

/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/* Header for class com_yunos_xulun_testcppjni_TestCppJni */

#ifndef _Included_com_yunos_xulun_testcppjni_TestCppJni
#define _Included_com_yunos_xulun_testcppjni_TestCppJni
#ifdef __cplusplus
extern "C" {
#endif
/*
 * Class:     com_yunos_xulun_testcppjni_TestCppJni
 * Method:    callCpp
 * Signature: ()I
 */
JNIEXPORT jint JNICALL Java_com_yunos_xulun_testcppjni_TestCppJni_callCpp
  (JNIEnv *, jclass);

#ifdef __cplusplus
}
#endif
#endif

然后我们写一个C++函数去实现它:

#include "com_yunos_xulun_testcppjni_TestCppJni.h"

JNIEXPORT jint JNICALL Java_com_yunos_xulun_testcppjni_TestCppJni_callCpp
  (JNIEnv *env, jclass obj){
    return (jint)0;
}

将头文件和cpp文件放到app/src/main/jni下面。

在Android工程中写个类去引用这个本地方法,就不多说了。编译运行,就生成了libtestcppjni.so。

还可以额外多设几个属性,比如编译选项,比如生成什么架构的库,引入什么库等等:

    android {
        compileSdkVersion 24
        buildToolsVersion "24.0.0"

        defaultConfig {
            applicationId "com.yunos.xulun.testcppjni"
            minSdkVersion 21
            targetSdkVersion 24
            versionCode 1
            versionName "1.0"
            ndk{
                moduleName "testcppjni"
                cFlags "-std=c++11"
                abiFilters "arm64-v8a","armeabi", "armeabi-v7a", "x86"
                ldLibs "log"
            }
        }
        buildTypes {
            release {
                minifyEnabled false
                proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
            }
        }
    }

新的实验性的编译系统的NDK支持

其实,新的实验性编译系统,对于java的编译也是变了不少。

下面,我们follow步骤去升级到新的编译系统:

升级gradle的版本

首先,要确保gradle的版本在2.10以上,我们通过修改gradle/wrapper下的gradle-wrapper.properties中的distributionUrl来实现这一点。

distributionUrl=https\://services.gradle.org/distributions/gradle-2.10-all.zip

升级plug-in

我们修改build.gradle,从正式版改成实验版,在我写这篇文件时,最新版本是0.8.0-alpah5。我们保守点,选个不带alpha的0.7.2:

buildscript {
    repositories {
        jcenter()
    }
    dependencies {
        classpath 'com.android.tools.build:gradle-experimental:0.7.2'

        // NOTE: Do not place your application dependencies here; they belong
        // in the individual module build.gradle files
    }
}

修改代码

经过上一步的修改,系统已经提示,无法找到我们之前用的"com.android.application"的包名了,所以我们要针对新的'com.android.model.applicaiton’包进行修改:

apply plugin: 'com.android.model.application'
model {
    android {
        compileSdkVersion 24
        buildToolsVersion "24.0.0"

        defaultConfig {
            applicationId "com.yunos.xulun.testcppjni"
            minSdkVersion.apiLevel 21
            targetSdkVersion.apiLevel 24
            versionCode 1
            versionName "1.0"
            buildConfigFields{
                    create() {
                        type "int"
                        name "VALUE"
                        value "1"
                    }
            }
            ndk {
                moduleName "testcppjni"
                cppFlags.add("-std=c++11")
                ldLibs.add("log")
            }
        }
        buildTypes {
            release {
                minifyEnabled false
                proguardFiles.add(file('proguard-rules.pro'))
            }
        }
        productFlavors{
            create("Flavor"){
                applicationId "com.yunos"
            }
        }
    }
}
dependencies {
    compile fileTree(dir: 'libs', include: ['*.jar'])
    testCompile 'junit:junit:4.12'
    compile 'com.android.support:appcompat-v7:24.0.0'
}

针对ndk部分,moduleName部分不用变。但是像cppFlags和ldLibs现在要通过add方法来进行添加了。

我们还可以加上调试和stl的支持:

apply plugin: 'com.android.model.application'
model {
    android {
        compileSdkVersion 24
        buildToolsVersion "24.0.0"

        defaultConfig {
            applicationId "com.yunos.xulun.testcppjni"
            minSdkVersion.apiLevel 21
            targetSdkVersion.apiLevel 24
            versionCode 1
            versionName "1.0"
            buildConfigFields{
                    create() {
                        type "int"
                        name "VALUE"
                        value "1"
                    }
            }
            ndk {
                moduleName "testcppjni"
                cppFlags.add("-std=c++11")
                ldLibs.add("log")
                stl "stlport_shared"
            }
        }
        buildTypes {
            release {
                minifyEnabled false
                proguardFiles.add(file('proguard-rules.pro'))
                ndk{
                    debuggable true
                }
            }
        }
        productFlavors{
            create("Flavor"){
                applicationId "com.yunos"
            }
        }
    }
}
dependencies {
    compile fileTree(dir: 'libs', include: ['*.jar'])
    testCompile 'junit:junit:4.12'
    compile 'com.android.support:appcompat-v7:24.0.0'
}

直接调用NDK或者CMake

上面这两种方式还不算完,生命不息,折腾不止,让我们一起跟随Android Studio继续折腾。
从2.2版本开始,Android Studio开始在64位OS上支持

这次我们直接调用NDK去build。
之前我们一直没有写Android.mk,现在要用NDK了,就写一个吧:

LOCAL_PATH := $(call my-dir)

include $(CLEAR_VARS)

LOCAL_MODULE    := testcppjni
LOCAL_SRC_FILES := test.cpp

LOCAL_CFLAGS += -std=c++11

LOCAL_ARM_MODE := thumb

include $(BUILD_SHARED_LIBRARY)

Application.mk也写个吧:

APP_ABI := armeabi armeabi-v7a arm64-v8a

然后改build.gradle:

android {
    compileSdkVersion 20
    buildToolsVersion "24.0.0"
    externalNativeBuild{
        ndkBuild{
            path "Android.mk"
        }
    }

    defaultConfig {
        applicationId "com.yunos.xulun.testcppjni"
        minSdkVersion 19
        targetSdkVersion 19
        externalNativeBuild {
            ndkBuild {
                targets "testcppjni"
                arguments "NDK_APPLICATION_MK:=Application.mk"
                cppFlags "-std=c++11"
                abiFilters "armeabi-v7a", "armeabi","arm64-v8a"
            }
        }
    }

综述

从目前的情况看,可以继续用过时的老方法,因为后两种都还没有正式发布。
有兴趣的可以跟进新的编译系统,升级到2.2之后,也可以考虑使用externalNativeBuild。

目录
相关文章
|
10天前
|
开发框架 前端开发 Android开发
安卓与iOS开发中的跨平台策略
在移动应用开发的战场上,安卓和iOS两大阵营各据一方。随着技术的演进,跨平台开发框架成为开发者的新宠,旨在实现一次编码、多平台部署的梦想。本文将探讨跨平台开发的优势与挑战,并分享实用的开发技巧,帮助开发者在安卓和iOS的世界中游刃有余。
|
15天前
|
搜索推荐 Android开发 开发者
探索安卓开发中的自定义视图:打造个性化UI组件
【10月更文挑战第39天】在安卓开发的世界中,自定义视图是实现独特界面设计的关键。本文将引导你理解自定义视图的概念、创建流程,以及如何通过它们增强应用的用户体验。我们将从基础出发,逐步深入,最终让你能够自信地设计和实现专属的UI组件。
|
17天前
|
Android开发 Swift iOS开发
探索安卓与iOS开发的差异和挑战
【10月更文挑战第37天】在移动应用开发的广阔舞台上,安卓和iOS这两大操作系统扮演着主角。它们各自拥有独特的特性、优势以及面临的开发挑战。本文将深入探讨这两个平台在开发过程中的主要差异,从编程语言到用户界面设计,再到市场分布的不同影响,旨在为开发者提供一个全面的视角,帮助他们更好地理解并应对在不同平台上进行应用开发时可能遇到的难题和机遇。
|
17天前
|
存储 API 开发工具
探索安卓开发:从基础到进阶
【10月更文挑战第37天】在这篇文章中,我们将一起探索安卓开发的奥秘。无论你是初学者还是有经验的开发者,这篇文章都将为你提供有价值的信息和建议。我们将从安卓开发的基础开始,逐步深入到更复杂的主题,如自定义组件、性能优化等。最后,我们将通过一个代码示例来展示如何实现一个简单的安卓应用。让我们一起开始吧!
|
18天前
|
存储 XML JSON
探索安卓开发:从新手到专家的旅程
【10月更文挑战第36天】在这篇文章中,我们将一起踏上一段激动人心的旅程,从零基础开始,逐步深入安卓开发的奥秘。无论你是编程新手,还是希望扩展技能的老手,这里都有适合你的知识宝藏等待发掘。通过实际的代码示例和深入浅出的解释,我们将解锁安卓开发的关键技能,让你能够构建自己的应用程序,甚至贡献于开源社区。准备好了吗?让我们开始吧!
26 2
|
19天前
|
Android开发
布谷语音软件开发:android端语音软件搭建开发教程
语音软件搭建android端语音软件开发教程!
|
Java Android开发
Android JNI开发从0到1,java调C,C调Java,保姆级教程详解
Android JNI开发从0到1,java调C,C调Java,保姆级教程详解
88 1
|
Java Android开发 C++
Android中的JNI开发,你了解多少?
Android中的JNI开发,你了解多少?
99 0
|
存储 缓存 Java
Android C++系列:JNI开发准则
JNI 定义了两个关键数据结构,即“JavaVM”和“JNIEnv”。两者本质上都是指向函数表的二级指针。(在 C++ 版本中,它们是一些类,这些类具有指向函数表的指针,并具有通过该函数表间接调用的 JNI 函数的成员函数。)JavaVM 提供“调用接口”函数,我们可以利用这些函数创建和销毁 JavaVM。理论上,每个进程可以有多个 JavaVM,但 Android 只允许有一个。
300 0
|
算法 Java 编译器
手把手教你如何在Android下进行JNI开发(入门)
手把手教你如何在Android下进行JNI开发(入门)
967 0
手把手教你如何在Android下进行JNI开发(入门)