Android NDK入門開發(一)

JNI(Java Native Interface):Java原生接口,是Java和其他原生代碼語言(例如 C 和 C++)通信的橋樑。
NDK(Native Development Kit):原生開發工具集,是一套允許您使用原生代碼語言(例如 C 和 C++)實現程序功能的工具集。
CMake:Android推薦使用的NDK構建工具,從AS 2.2版本之後開始支持(包含2.2版本)。

環境搭建:
首先先創建一個工程,因爲我的AS是最新版本的,可能創建的方式會跟舊版的稍有不同

在這裏插入圖片描述
點擊Native C++ ,然後Next,再Next,這裏是C++最低支持的NDK版本,這裏就默認了,最後再finsh,一個工程就建好了。
在這裏插入圖片描述
工程建好了,但是你的AS必須要有NDK才能跑的起程序,那就下來就是下載NDK的一些工具
在這裏插入圖片描述
主要是下載這三個就可以了。
還有,別忘了要在這選上下載好的NDK,沒有下載的話就下載一個,不然會報錯在這裏插入圖片描述
最後Sync Project with gradle Files,一個NDK的工程就這麼創建好了,接下來我們看一下工程的目錄
跟以往的工程不同之處是多了些配置文件,例如CMakeLists.txt等,我們一個個說
在這裏插入圖片描述
首先是build.gradle文件,看看源碼


apply plugin: 'com.android.application'


android {
    compileSdkVersion 29
    buildToolsVersion "29.0.2"


    defaultConfig {
        applicationId "com.example.myapplication"
        minSdkVersion 24
        targetSdkVersion 29
        versionCode 1
        versionName "1.0"


        testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner"


        externalNativeBuild {
            cmake {
                cppFlags ""
            }
        }
    }


    buildTypes {
        release {
            minifyEnabled false
            proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro'
        }
    }


    externalNativeBuild {
        cmake {
            path "src/main/cpp/CMakeLists.txt"
            version "3.10.2"
        }
    }
}


dependencies {
    implementation fileTree(dir: 'libs', include: ['*.jar'])


    implementation 'androidx.appcompat:appcompat:1.1.0'
    implementation 'androidx.constraintlayout:constraintlayout:1.1.3'
    testImplementation 'junit:junit:4.12'
    androidTestImplementation 'androidx.test.ext:junit:1.1.1'
    androidTestImplementation 'androidx.test.espresso:espresso-core:3.2.0'
}

可以看到build.gradle配置中多了兩個externalNativeBuild配置項,defaultConfig裏面的:主要配置了Cmake的命令參數,而defaultConfig外面的:主要定義了CMake的構建腳本CMakeLists.txt的路徑。 CMakeLists.txt是CMake的構建腳本,作用相當於ndk-build中的Android.mk

CMakeLists.txt

# 設置Cmake最小版本
cmake_minimum_required(VERSION 3.4.1)


# 編譯library
add_library( # 設置library名稱
             native-lib

             # 設置library模式
             # SHARED模式會編譯so文件,STATIC模式不會編譯
             SHARED

             # 設置原生代碼路徑
             src/main/cpp/native-lib.cpp )

# 定位library
find_library( # library名稱
              log-lib

              # 將library路徑存儲爲一個變量,可以在其他地方用這個變量引用NDK庫
              # 在這裏設置變量名稱
              log )

# 關聯library
target_link_libraries( # 關聯的library
                       native-lib


                       # 關聯native-lib和log-lib
                       ${log-lib} )

原生代碼native-lib.cpp( 方法名是通過 Java_包名_類名_方法名 的方式命名的)

#include <jni.h>
#include <string>


extern "C" JNIEXPORT jstring JNICALL
Java_com_example_myapplication_MainActivity_stringFromJNI(
        JNIEnv* env,
        jobject /* this */) {
    std::string hello = "Hello from C++";
    return env->NewStringUTF(hello.c_str());
}

看下MainActivity

package com.example.myapplication;


import androidx.appcompat.app.AppCompatActivity;


import android.os.Bundle;
import android.widget.TextView;


public class MainActivity extends AppCompatActivity {


    // Used to load the 'native-lib' library on application startup.
    static {
        System.loadLibrary("native-lib");
    }


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


        // Example of a call to a native method
        TextView tv = findViewById(R.id.sample_text);
        tv.setText(stringFromJNI());
    }


    /**
     * A native method that is implemented by the 'native-lib' native library,
     * which is packaged with this application.
     */
    public native String stringFromJNI();
}

這裏主要是一個文本框,然後調用C/C++的方法,輸出Hello from C++,運行後的效果圖如下
在這裏插入圖片描述

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章