Android JNI概述

本文基于Android 9.0源碼分析

Android JNI簡介

JNI是Java Native Interface, 它提供了一種從字節碼(Java/Kotlin)到Native代碼(c/c++/assembly)的交互方式

JavaVM與JNIEnv

JNI定義了兩個關鍵的數據結構:JavaVM和JNIEnv

  • JavaVM
    • JavaVM提供了"invocation interface"函數表,允許你創建和銷毀JavaVM,理論上你可以在進程內創建多個JavaVM實例,但Android只允許創建一個。
  • JNIEnv
    • JNIEnv提供了大部分JNI方法,JNIEnv存儲在TLS中,基于上述原因,不能在線程間共享JNIEnv。上面的圖中JNINativeInterface實際是全局結構體,圖有點問題。
  • 線程
    • Android中的線程都是Linux線程,由Linux內核調度。通常通過Thread.start()啟動線程,但是也可以使用其他方法啟動線程,然后attach到JavaVM。比如,使用pthread_create()創建線程,調用AttachCurrentThread()或者AttachCurrentThreadAsDaemon()attach到JavaVM。注意,在attach之前,沒有JNIEnv,無法使用JNI。

Android Native方法注冊

使用RegisterNatives顯示注冊

示例

public class MainActivity extends AppCompatActivity {

    // Used to load the 'native-lib' library on application startup.
    static {
        System.loadLibrary("native-lib");
    }
    ......
    /**
     * A native method that is implemented by the 'native-lib' native library,
     * which is packaged with this application.
     */
    public native String stringFromJNI();
}
// 1) using RegisterNatives register native method
static jstring StringFromJNI(JNIEnv *env, jobject obj) {
  std::string hello = "Hello from C++";
  return env->NewStringUTF(hello.c_str());
}

static JNINativeMethod jni_methods[] = {
    {"stringFromJNI", "()Ljava/lang/String;", (void *)StringFromJNI}
};

// 這里JNI_OnLoad不需要添加extern "C"
JNIEXPORT jint JNI_OnLoad(JavaVM* vm, void* reserved) {
  JNIEnv* env;

  if (vm->GetEnv(reinterpret_cast<void**>(&env), JNI_VERSION_1_6) != JNI_OK) {
    return -1;
  }

  if (env->RegisterNatives(env->FindClass("lbtrace/jniregister/MainActivity"),
      jni_methods, sizeof(jni_methods) / sizeof(JNINativeMethod)) != JNI_OK)
    return -1;

  return JNI_VERSION_1_6;
}

原理

  • 首先獲取調用者的類加載(這里是應用類加載器),然后在類加載器的DexPathList中查找Native庫

    • 根據Native庫的名字產生平臺相關的庫名
    • 在DexPathList的Native庫路徑列表中查找Native庫,找到后返回庫的絕對路徑
  • JavaVM負責加載Native庫,如果沒有加載過,裝載Native庫到進程地址空間,然后查找Native庫中是否有函數JNI_OnLoad(),如果存在,執行;否則運行時動態查找Native方法。

  • JNI_OnLoad()中,首先獲取當前線程的JNIEnv,然后調用其RegisterNatives()注冊Native方法。

    • 首先檢查需要注冊Native方法的信息是否合法
    • 根據方法名、方法簽名查找對應的Native方法的ArtMethod
    • 將Native函數地址寫入ArtMethod對象特定的成員中

運行時動態查找

示例

// 2) Android Runtime dynamic find native method
// 必須添加extern "C"
extern "C" JNIEXPORT jstring JNICALL
Java_lbtrace_jniregister_MainActivity_stringFromJNI(
        JNIEnv* env,
        jobject /* this */) {
    std::string hello = "Hello from C++";
    return env->NewStringUTF(hello.c_str());
}

原理

  • ART運行時執行到調用Native方法的字節碼時,首先獲取方法對應的ArtMethod,根據ArtMethod對象的地址以及成員entry_point_from_quick_compiled_code_的偏移得到JNI Stub的地址。
  • 在JNI Stub函數中,首先進行Native方法調用前的準備工作(參數處理、進程狀態切換等),然后根據特定的Native方法名(java_xxx_xxx_xxx)查找Native方法,調用Native方法,Native方法地址保存到ArtMethod,最后是Native方法調用結束后的清理工作。

思考:Native方法中jobject類型的參數是什么?

在先前的Android版本中是Local Reference

  • StackReference

Android JNI本地引用和全局引用

對于基本類型,進行JNI調用時,可以直接拷貝到Native方法,但是對于Java對象采用引用傳遞。虛擬機必須能夠追蹤到所有傳遞到Native方法的Java對象,避免被GC回收。當Native方法不再需要Java對象時,必須有一種方法通知虛擬機。

對于本地引用及全局引用的實現細節,將在后續文章中討論。

Local Reference

本地引用在Native方法調用期間可用,在Native方法返回后自動釋放。

  • 對于大的Java對象的本地引用,不用時應及時釋放
  • 不要創建太多本地應用

Global Reference

全局引用不會自動釋放,直到顯式的刪除

Weak Global Reference

弱全局引用是一種特殊的全局引用,與普通的全局引用不同的是,GC可以回收弱全局引用關聯的Java對象。當GC運行時,會回收一個僅僅被弱全局引用關聯Java對象。所以在使用之前,應該首先檢查弱全局引用的Java對象是否被回收。

思考:Android中Java引用到底是什么?

  • 根據Java引用原理,可以在Native層直接修改Java對象內容
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) {
        ......
        TestObject testObject = new TestObject();
        Log.i("JNI", "Before : " + testObject.month + "月" +
                testObject.day + "日");
        stringFromJNI(testObject);
        Log.i("JNI", "After : " + testObject.month + "月" +
                testObject.day + "日");
        ......
    }

    public native String stringFromJNI(Object obj);

    static class TestObject {
        int month = 11;
        int day = 29;
    }
}

extern "C" JNIEXPORT jstring JNICALL
Java_lbtrace_jniregister_MainActivity_stringFromJNI(
        JNIEnv* env,
        jobject obj, jobject test_obj) {
    std::string hello = "Hello from C++";

    // Java TestObject Mirror address
    int32_t *test_obj_ptr = reinterpret_cast<int32_t *>(
        *(reinterpret_cast<int32_t *>(test_obj)));

    // According TestObject memory layout
    // Swap month field and day field in TestObject
    int32_t tmp = *(test_obj_ptr + 2);
    *(test_obj_ptr + 2) = *(test_obj_ptr + 3);
    *(test_obj_ptr + 3) = tmp;

    return env->NewStringUTF(hello.c_str());
}

參考

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 227,428評論 6 531
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,024評論 3 413
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 175,285評論 0 373
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,548評論 1 307
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,328評論 6 404
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 54,878評論 1 321
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 42,971評論 3 439
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,098評論 0 286
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 48,616評論 1 331
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,554評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,725評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,243評論 5 355
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 43,971評論 3 345
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,361評論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,613評論 1 280
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,339評論 3 390
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 47,695評論 2 370

推薦閱讀更多精彩內容

  • 閑來翻譯了一篇官方的JNI Tips,網上看到的翻譯版本要么是時間久了不同步了,要么翻譯的過于生硬,看得我懷疑自己...
    生活簡單些閱讀 1,771評論 1 4
  • 本系列文章如下: Android JNI(一)——NDK與JNI基礎Android JNI學習(二)——實戰JNI...
    隔壁老李頭閱讀 205,259評論 25 534
  • 用兩張圖告訴你,為什么你的 App 會卡頓? - Android - 掘金 Cover 有什么料? 從這篇文章中你...
    hw1212閱讀 12,765評論 2 59
  • 在分析IPC基于Android 6.0)的過程中,里面的核心部分是Native的,并且還包含一些linux ker...
    Sophia_dd35閱讀 3,397評論 0 7
  • 準備明天的考試。 難過。 丑陋似乎變成了一種原罪。 從來就沒有過自信。 也沒努力過,小時候似乎有一點打扮,照照鏡子...
    聽雷雷說閱讀 131評論 0 0