Android Dev Intro - Android NDK with Android Studio


https://shaneenishry.com/blog/2014/08/17/ndk-with-android-studio/


Unless you are using a game engine with an integrated scripting language, game development is usually done with C++. On the other hand, most Android applications are developed with Java.

This post will explain how to integrate C++ code into Android Studio, use gradle to build it and clear out some misconceptions.

If you used Eclipse you probably know there was a relatively easy way to add "native library" support to your project but now that most projects are being moved to the newer Android Studio it seems like there is no more C++ support. That is incorrect.

There are two ways to add C++ code code into your project. The first is by including prebuilt libraries which you have compiled elsehwere or were pre-compiled for you. The other is to compile them directly from your Android Studio project.

Including pre-built C++ libraries

Including pre-built libraries is simple. Android Studio will look for a specific directory tree and copy all libraries it finds.

The folder structure should be in your Android Studio module's folder and look like this

  • [module_name]
  • * [src]
  • * * [main]
  • * * * [jniLibs]
  • * * * * [armeabi]
  • * * * * [armeabi-v7a]
  • * * * * [x86]
  • * * * * [mips]

Make sure the folder tree is located inside your module, not inside your project root. For example **[projectroot]/app/...**_

The names of the end folders are the same as the ABI the library was compiled for. Make sure to add new ones when you compile for arm64-v8ax86-64 and mips64.

This is how it looks like from Android Studio:

Now you can load your shared libraries (.so) from Android using

String libName = "helloNDK"; // the module name of the library, without .so
System.loadLibrary( libName );

Note that compiling a shared library with the ndk generatres a lib + yourModuleName + .so. When loading your library you should only call your module's name, without the prefix and suffix.

Changing the jniLibs folder

If for some reason you want your C++ libraries path to be somewhere else, you can change it by settingsourceSets.main.jniLibs.srcDirs in your module's build.gradle:

android {

    // .. android settings ..

    sourceSets.main {
        jniLibs.srcDir 'src/main/myCppLibraries' // <-- Set your folder here!
    }
}

Compiling C++ in Android Studio

Adding precompiled libraries is easy but tabbing between terminal and Android Studio can be annoying. If you are developing your C++ code with Android Studio and want it to be compiled when you build your .apk this is how you can do it.

There are 3 things you need to do:

  1. Add ndk.dir variable inside the local.properties file.
  2. Configure ndk module with gradle.
  3. Add C++ files to the expected folder.
  4. (Optional) Set product flavors.
Adding ndk.dir variable

The first thing to do is to configure the path to your ndk folder. Inside your proejct's root folder there should be a file called local.properties. Add a ndk.dir variable directing to your ndk folder. For example:

sdk.dir=/Users/shanee/Development/Android/sdk
ndk.dir=/Users/shanee/Development/ndk

That was easy!

Configuring ndk with gradle

Ok, now to tell gradle to compile our C++ code.

Inside your module's build.gradle file locate your android's defaultConfig section and add the following:

ndk {
    moduleName "moduleNameHere"
}

Make sure to replace the string with your module's name!

Here is mine for reference:

android {
    compileSdkVersion 19
    buildToolsVersion "20.0.0"

    defaultConfig {
        applicationId "com.example.ndksample"
        minSdkVersion 9
        targetSdkVersion 19
        versionCode 1
        versionName "1.0"

        ndk {
            moduleName "myEpicGameCode" // <-- This is the name of my C++ module!
        }
    }

    // ... more gradle stuff here ...

} // end of android section

Note that you can also add additional C++ configuration such as cFlagsstl and ldLibs.

ndk {
    moduleName "myEpicGameCode"
    cFlags "-DANDROID_NDK -D_DEBUG DNULL=0" // Define some macros
    ldLibs "EGL", "GLESv3", "dl", "log"         // Link with these libraries!
    stl "stlport_shared"                        // Use shared stlport library
}
Adding C++ source files

Now that Android Studio and Gradle are configured to compile C++ source code it is time to feed them some files!

Naturally, Gradle expects the C++ files to be inside [module]/src/main/jni/. Simply adding your files there will do the trick:

The next time you try to run the application, gradle will build the C++ module for you. It will even shout at you for C++ errors in the gradle console, for example I added this bad_function:

void bad_function()
{
    int myInt = 4;
    myInt = 10
}

And gradle responded with:

Changing the folder of your source files

Just like with the jniLibs folder, you can tell gradle to look for your C++ files in a different path. I for example prefer my source code inside source. I can set it by editing the sourceSets.main.jni.srcDirs variable:

android {

    // .. android settings ..

    sourceSets.main {
        jni.srcDirs 'src/main/source'
    }
}
(Optional) Setting product flavors

You can set different compilation variables for different platforms. For example you might want to define a different macro for x86 and arm to use different optimizations or to include different header files.

You can do this by adding the productFlavors section under android in the module's build.gradle file.

The most obvious setting would be to avoid compiling the library for all platforms when building the .apk for a specific ABI. For that you can use the abiFilter variable:

android {

    // .. android settings ..

    productFlavors {
        x86 {
            ndk {
                abiFilter "x86"
            }
        }
        arm {
            ndk {
                abiFilter "armeabi-v7a"
            }
        }
        mips {
            ndk {
                abiFilter "mips"
            }
        }
    }

} // android

The next time you build your project for a specific ABI target the ndk will know which build type it should create and not to use the default abi settings which is to compile for all platforms.

You can also override the default cFlags and other settings here, just add them inside the ndk section under the build of your choice.

Summary

I oriented this guide toward people who are looking to convert their project from Eclipse to Android Studio and omitted explanation on how to use JNI and call C++ functions from Java/Android. If there is demand for a JNI tutorial I will write a post for it.

Hope this post was helpful for you! Keep making awesome games for Android :)


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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值