Android的JNI調用

来源:http://www.cnblogs.com/jiayayao/archive/2017/06/24/7073513.html
-Advertisement-
Play Games

Android提供NDK開發包來提供Android平臺的C++開發,用來擴展Android SDK的功能。主要包括Android NDK構建系統和JNI實現與原生代碼通信兩部分。 一、Android NDK構建系統 1.1 構建庫 Android NDK的構建系統是基於GNU Make的。Andro ...


  Android提供NDK開發包來提供Android平臺的C++開發,用來擴展Android SDK的功能。主要包括Android NDK構建系統和JNI實現與原生代碼通信兩部分。

一、Android NDK構建系統

 1.1 構建庫

   Android NDK的構建系統是基於GNU Make的。Android GNU Make系統除了需要一些內部的GNU片段外,還需要兩個文件:Android.mk和Application.mk。Android NDK源碼給了很多的例子,以HelloJni為例,Android.mk源碼: 

#Android.mk必須以LOCAL_PATH變數開頭
LOCAL_PATH := $(call my-dir) #清除除了LOCAL_PATH以外的LOCAL_<name>變數,例如LOCAL_MODULE與LOCAL_SRC_FILES等 include $(CLEAR_VARS) #每一個原生組件被稱為一個模塊 LOCAL_MODULE := hello-jni
#源文件 LOCAL_SRC_FILES :
= hello-jni.c #編譯為共用庫,即尾碼名為.so include $(BUILD_SHARED_LIBRARY)

  Application.mk源碼: 

 #一般選擇APP_ABI := armeabi-v7a就夠了
 APP_ABI := all

  為了建立可供主應用程式使用的模塊,必須將該模塊變成共用庫。按照上述必不可少的步驟,可以繼續編譯多個共用庫。

  Android也可以編譯靜態庫(尾碼名為.a),但是實際的Android應用程式並不直接使用靜態庫,並且應用程式包中也不包含靜態庫。靜態庫可以用來構建共用庫。但是,當靜態庫與多個共用庫相連時,應用程式包中會包含靜態庫的多個副本,徒增應用程式包的大小。這種情況下,可以不構建靜態庫,而是將通用模塊作為共用庫建立起來,動態連接依賴模塊以消除重覆的副本。如下Android.mk實現的是共用庫之間的代碼共用。 

LOCAL_PATH := $(call my-dir)

#第三方AVI庫
include $(CLEAR_VARS)
LOCAL_MODULE    := avilib
LOCAL_SRC_FILES := avilib.c
include $(BUILD_SHARED_LIBRARY)

#原生模塊1
include $(CLEAR_VARS)
LOCAL_MODULE    := module1
LOCAL_SRC_FILES := module1.c

LOCAL_SHARED_LIBRARIES := avilib
include $(BUILD_SHARED_LIBRARY)

#原生模塊2
include $(CLEAR_VARS)
LOCAL_MODULE    := module2
LOCAL_SRC_FILES := module2.c

LOCAL_SHARED_LIBRARIES := avilib
include $(BUILD_SHARED_LIBRARY) 

1.2 Prebuilt庫

  共用模塊編譯時要求有源代碼,為此Android提供了Prebuilt庫,以下場合,Prebuilt庫是非常有用的:

  1. 想在不發佈源代碼的情況下將你的模塊發佈給他人;
  2. 想使用共用模塊的預建版來加速構建過程。

  其他構建系統變數:

  LOCAL_CFLAGS:一組可選的編譯器標誌,在編譯C和C++源文件的時候會被傳送給編譯器;

  LOCAL_CPP_FLAGS:一組可選的編譯器標誌,在只編譯C++源文件時被傳送給編譯器;

  LOCAL_LDLIBS:鏈接標誌的可選列表,它主要用於傳送要進行動態鏈接的系統庫列表。如鏈接日誌庫:

  LOCAL_LDFLAGS += -llog 

  APP_CPPFLAGS:編譯器標誌,在編譯任何模塊的C++源文件時這些標誌都會被傳送給編譯器。 

  nkd-build腳本命令:

ndk-build –C /project path
ndk-build –B
ndk-build clean 

二、JNI實現與原生代碼通信 

  Java層代碼如下: 

public class HelloJni extends Activity
{
    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState)
    {
        super.onCreate(savedInstanceState);

        TextView  tv = new TextView(this);
        tv.setText( stringFromJNI() );
        setContentView(tv);
    }

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

    /* This is another native method declaration that is *not*
     * implemented by 'hello-jni'. This is simply to show that
     * you can declare as many native methods in your Java code
     * as you want, their implementation is searched in the
     * currently loaded native libraries only the first time
     * you call them.
     *
     * Trying to call this function will result in a
     * java.lang.UnsatisfiedLinkError exception !
     */
    public native String  unimplementedStringFromJNI();

    /* this is used to load the 'hello-jni' library on application
     * startup. The library has already been unpacked into
     * /data/data/com.example.hellojni/lib/libhello-jni.so at
     * installation time by the package manager.
     */
    static {
        System.loadLibrary("hello-jni");
    }
}  

  JNI代碼如下,其中Java方法stringFromJNI不帶任何參數,但是原生方法帶兩個參數:

jstring
Java_com_example_hellojni_HelloJni_stringFromJNI( JNIEnv* env,
                                                  jobject thiz )
{
    return (*env)->NewStringUTF(env, "Hello from JNI !");
}

  第一個參數JNIEnv是指向可用JNI函數表的藉口指針;第二個參數jobjects是HelloJni類實例的Java對象引用。  

  這裡註意C與C++代碼稍有不同,C代碼如下: 

  return (*env)->NewStringUTF(env, "Hello from JNI ! "); 

  C++代碼如下: 

  return env->NewStringUTF("Hello from JNI ! "); 

  這是因為C++代碼中,JNIEnv實際上是一個C++類實例,JNI函數以成員函數的形式存在,因此JNI方法調用不要求JNIEnv實例作參數。

2.1 C/C++ 頭文件生成器:javah

  JDK自帶一個名為javah的命令行工具,該工具由Java類文件的原始定義生成原生函數名及其參數列表,這樣程式員避免編寫繁雜多餘的定義。C/C++源文件只需要包含這個頭文件並提供原生方法實現。 

  javah的參數列表如下: 

C:\Users\jiayayao>javah
用法:
  javah [options] <classes>
其中, [options] 包括:
  -o <file>                輸出文件 (只能使用 -d 或 -o 之一)
  -d <dir>                 輸出目錄
  -v  -verbose             啟用詳細輸出
  -h  --help  -?           輸出此消息
  -version                 輸出版本信息
  -jni                     生成 JNI 樣式的標頭文件 (預設值)
  -force                   始終寫入輸出文件
  -classpath <path>        從中載入類的路徑
  -cp <path>               從中載入類的路徑
  -bootclasspath <path>    從中載入引導類的路徑
<classes> 是使用其全限定名稱指定的
(例如, java.lang.Object)。

  生成頭文件的命令行參數如下:

 javah -classpath bin/classes com.example.hellojni.Hellojni  

2.2 數據類型

Java有兩種數據類型:基本數據類型和引用數據類型:基本數據類型中Java/JNI/C++的映射關係如下:

   引用類型的類型映射關係如下: 

  引用類型以不透明的引用方式傳遞給原生代碼,而不是以原生數據類型的的形式呈現,因此引用類型不能直接使用和修改。JNI提供了與這些引用類型密切相關的一組API。

  字元串操作: 

// 創建字元串
jstring javaString;
javaString = (*env)->NewStringUTF(env, "hello world!");
// 記憶體溢出時,會返回NULL,註意判空

// 將Java字元串轉換成C字元串
const jbyte* str;
jboolean isCopy;
str = (*env)->GetStringUTFChars(env, javaString, &isCopy);
if (0 != str) {

}

// 釋放字元串
(*env)->ReleaseStringUTFChars(env, javaString, str);

   數組操作: 

// 創建數組
jintArray javaArray;
javaArray = (*env)->NewIntArray(env, 10);
if(0!=javaArray) {
    
}

// Get<Type>ArrayRegion函數將給定的基本Java數組複製到給定的C數組中
// 將Java數組區複製到C數組中 jint nativeArray[10]; (*env)->GetIntArrayRegion(env, javaArray, 0, 10, nativeArray); // 從C數組向Java數組提交所做的修改 (*env)->SetIntArrayRegion(env, javaArray, 0, 10, nativeArray);

   原生方法的記憶體分配超出了虛擬機的管理範圍,且不能用虛擬機的垃圾回收器回收原生方法中的記憶體。

  原生代碼回到Java損耗性能,建議將所有需要的參數傳遞給原生代碼調用,而不是讓原生代碼回到Java中。

  先記錄這麼多,以後接著補充。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 本來上個月就像發的,但是一直忙啊忙的也沒時間整理,所以拖到了現在。 好吧上面這句就是廢話,我就是感概下。下麵是正文。 前段時間在弄一個輕量級的web項目,要構建一個樹狀結構目錄,同時希望能把部分選中的狀態給表現出來。項目中只用了jquery,個人也不想再引入一些其他的js框架或者插件,一個是考慮到界 ...
  • 記錄JavaScript練習筆記和知識點整理的第二天,遇到在程式里使用等差數列求和公式的操作!遇到使用正則表達式判斷數組中是否有重覆的值!又學會了新操作。 ...
  • var arr1 = [1,2,3,4,5,6]; arr1[0]; arr1[1]; console.log("長度:"+arr1.length); 一、遍曆數組 / * i=下標*/ for(var i=0;i<arr1.length;i++){ console.log(arr1[i]); } ...
  • Es6中如何使用splic,delete等數組刪除方法 1:js中的splice方法 splice(index,len,[item]) 註釋:該方法會改變原始數組。 splice有3個參數,它也可以用來替換/刪除/添加數組內某一個或者幾個值 index:數組開始下標 len: 替換/刪除的長度 it ...
  • 在《vue-cli搭建的項目中增加後臺mock介面》中實現了後臺mock,但是前端post的t數據都要在mock的後臺介面中使用req的接收數據事件獲取http協議body中的數據。 如果前端需要使用cookie,後端要讀取,那麼在後臺mock的介面中還要獲取req的headers,並從中取得coo ...
  • 為 device node 取 label name, 可以在其它位置使用 &label 存取 device node。 Ex ./arch/arm/boot/dts/stm32f429.dtsi ./arch/arm/boot/dts/stm32f429 disco.dts ...
  • 目錄 RunTime 概述 RunTime消息機制 RunTime交換方法 RunTime消息轉發 RunTime關聯對象 RunTime實現字典與模型互轉 1.RunTime 概述 我們在面試的時候,經常都會被問到這麼個問題:為什麼說OC是一門動態的語言???其實也就是想知道你對runtime的了 ...
  • 問題:E/NotificationService: Not posting notification with icon==0: Notification(pri=0 contentView=null vibrate=null sound=null defaults=0x0 flags=0x0 co ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...