成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

研究了一下Android JNI,有幾個知識點不太懂

移動開發 Android
以前版本的Java線程不是OS線程,是JVM構造的用戶態線程(Green Thread),不能充分利用CPU,后期已經更改為使用OS線程實現。

[[437215]]

本文轉載自微信公眾號「程序喵大人」,作者程序喵大人 。轉載本文請聯系程序喵大人公眾號。

Java線程與Native(OS)線程的區別

聯系:Java線程其實是一層OS線程的封裝,本質上就是OS線程。【以前版本的Java線程不是OS線程,是JVM構造的用戶態線程(Green Thread),不能充分利用CPU,后期已經更改為使用OS線程實現。】【參考https://mp.weixin.qq.com/s/Gxqnf5vjyaI8eSYejm7zeQ】

區別:

Java線程可以直接拿到JNIEnv,OS線程需要先attach到JVM,才可以拿到JNIEnv。【個人理解區別在于是否attach了JVM】

  1. jint AttachCurrentThread(JavaVM *vm, void **p_env, void *thr_args); 

Java線程可以FindClass成功,OS線程則FindClass失敗,原因是兩者的ClassLoader不同,OS線程AttachCurrentThread后持有的ClassLoader是系統的ClassLoader,如果想要FindClass成功,需要在JNI_Onload時獲取一份當前庫的ClassLoader保存起來,下次FindClass時使用此ClassLoader去操作。

  1. static jobject g_class_loader = NULL
  2. static jmethodID g_find_class_method = NULL
  3. void on_load() { 
  4.     JNIEnv *env = get_jni_env(); 
  5.     if (!env) { 
  6.         return
  7.     } 
  8.     jclass capture_class = (*env)->FindClass(env, "com/captureandroid/BMMCaptureEngine"); 
  9.     jclass class_class = (*env)->GetObjectClass(env, capture_class); 
  10.     jclass class_loader_class = (*env)->FindClass(env, "java/lang/ClassLoader"); 
  11.     jmethodID class_loader_mid = (*env)->GetMethodID(env, class_class, "getClassLoader""()Ljava/lang/ClassLoader;"); 
  12.     jobject local_class_loader = (*env)->CallObjectMethod(env, capture_class, class_loader_mid); 
  13.     g_class_loader = (*env)->NewGlobalRef(env, local_class_loader); 
  14.     g_find_class_method = 
  15.         (*env)->GetMethodID(env, class_loader_class, "findClass""(Ljava/lang/String;)Ljava/lang/Class;"); 
  16.  
  17. jclass find_class(const char *name) { 
  18.     JNIEnv *env = bmm_util_get_jni_env(); 
  19.     if (!env) { 
  20.         return NULL
  21.     } 
  22.     jclass ret = (*env)->FindClass(env, name); 
  23.     jthrowable exception = (*env)->ExceptionOccurred(env); 
  24.     if (exception) { 
  25.         (*env)->ExceptionClear(env); 
  26.         jstring name_str = (*env)->NewStringUTF(env, name); 
  27.         ret = (jclass)(*env)->CallObjectMethod(env, g_class_loader, g_find_class_method, name_str); 
  28.         (*env)->DeleteLocalRef(env, name_str); 
  29.     } 
  30.     return ret; 

JNI的作用

貼出別人翻譯的【官方文檔https://docs.oracle.com/javase/7/docs/technotes/guides/jni/spec/design.html#wp16696】的一段話:

JNI最重要的設計目標就是在不同操作系統上的JVM之間提供二進制兼容,做到一個本地庫不需要重新編譯就可以運行不同的系統的JVM上面。為了達到這一點兒,JNI設計時不能關心JVM的內部實現,因為JVM的內部實現機制在不斷地變,而我們必須保持JNI接口的穩定。JNI的第二個設計目標就是高效。我們可能會看到,有時為了滿足第一個目標,可能需要犧牲一點兒效率,因此,我們需要在平臺無關和效率之間做一些選擇。最后,JNI必須是一個完整的體系。它必須提供足夠多的JVM功能讓本地程序完成一些有用的任務。JNI不能只針對一款特定的JVM,而是要提供一系列標準的接口讓程序員可以把他們的本地代碼庫加載到不同的JVM中去。有時,調用特定JVM下實現的接口可以提供效率,但更多的情況下,我們需要用更通用的接口來解決問題。

JNIEnv和JavaVM

就是個函數指針。

下圖是JNIEnv的指針結構:

JNIEnv其實是一個指向本地線程數據的接口指針,指針里面包含指向函數接口的指針,每一個接口函數在這表中都有一個預定義的偏移位置,類似C++虛函數表。

代碼如下:

  1. typedef const struct JNINativeInterface *JNIEnv;  
  2.  
  3. struct JNINativeInterface { 
  4.     void*       reserved0; 
  5.     void*       reserved1; 
  6.     void*       reserved2; 
  7.     void*       reserved3; 
  8.  
  9.     jint        (*GetVersion)(JNIEnv *); 
  10.  
  11.     jclass      (*DefineClass)(JNIEnv*, const char*, jobject, const jbyte*, 
  12.                         jsize); 
  13.     jclass      (*FindClass)(JNIEnv*, const char*); 
  14.     jobject     (*AllocObject)(JNIEnv*, jclass); 
  15.     jobject     (*NewObject)(JNIEnv*, jclass, jmethodID, ...); 
  16.     jobject     (*NewObjectV)(JNIEnv*, jclass, jmethodID, va_list); 
  17.     jobject     (*NewObjectA)(JNIEnv*, jclass, jmethodID, const jvalue*); 
  18.     ... 
  19. }; 
  20. JavaVM類似 
  21. struct JNIInvokeInterface { 
  22.     void*       reserved0; 
  23.     void*       reserved1; 
  24.     void*       reserved2; 
  25.  
  26.     jint        (*DestroyJavaVM)(JavaVM*); 
  27.     jint        (*AttachCurrentThread)(JavaVM*, JNIEnv**, void*); 
  28.     jint        (*DetachCurrentThread)(JavaVM*); 
  29.     jint        (*GetEnv)(JavaVM*, void**, jint); 
  30.     jint        (*AttachCurrentThreadAsDaemon)(JavaVM*, JNIEnv**, void*); 
  31. }; 
  32.  
  33. typedef const struct JNIInvokeInterface* JavaVM; 

知識點1:為什么使用函數表而不是寫死某些函數項?

可將JNI命名空間與本地代碼分離,一個虛擬機可以提供多個版本的JNI函數表,用于不同場景。例如,虛擬機可支持兩種JNI函數表:

一個用于調試,做較多的錯誤檢查。

一個用于發布,做較少的錯誤檢查,更高效。

知識點2:JNIEnv是thread-local,只在當前線程有效,Native方法不能將JNIenv從當前線程傳遞到另一個線程。不能跨線程使用JNIEnv【至于JNIEnv為什么設計成thread-local,沒搞明白】。

知識點3:線程間雖然不共享JNIEnv,但是共享JavaVM,然后可以通過GetEnv獲取到當前線程的JNIEnv。

jint GetEnv(JavaVM *vm, void **env, jint version);

知識點4:Native方法接收JNI接口指針作為參數。虛擬機保證在同一個線程傳入Native方法的是相同的JNIEnv。如果不同線程調用Native方法,傳入他們的JNIEnv不同。但JNIEnv間接指向的函數表在多個線程間是共享的。

知識點5:為什么在C語言中調用Native方法需要將JNIEnv當作參數傳遞,而C++中卻不需要?

  1. // C語言 
  2. jstring model_path = (*env)->NewStringUTF(env, path); 
  3. // C++ 
  4. jstring model_path = env->NewStringUTF(path); 

前面列出的JNIEnv是C語言形式,Java還單獨為C++封裝了一層JNIEnv,簡化版代碼:

  1. struct _JNIEnv { 
  2.     /* do not rename this; it does not seem to be entirely opaque */ 
  3.     const struct JNINativeInterface* functions; 
  4.  
  5. #if defined(__cplusplus) 
  6.  
  7.     jint GetVersion() 
  8. return functions->GetVersion(this); } 
  9.  
  10.     jclass FindClass(const charname
  11. return functions->FindClass(this, name); } 
  12. #endif 

其實本質上還是調用的C語言那種形式的接口。

JNI中數據如何傳遞

這里不詳細介紹了,大體就是int,float這種基本類型采用拷貝,對象和byte數組等使用引用形式,所以其實Java層的byte字節流數據傳到Native層基本不耗時,不會發生拷貝【但是Native層如果想使用持有這塊數據,那就得自己拷貝一份了】。

還有些GlobalReference、LocalReference以及為什么要Delete LocalReference的這類知識點,這些比較基礎,就不介紹了,估計大家也都懂。

推薦閱讀

https://www.cnblogs.com/kexinxin/p/11689641.html

ndk官方文檔

https://developer.android.com/ndk/guides

參考資料

http://luori366.github.io/JNI_doc/jni_design_theory.html

https://www.cnblogs.com/kexinxin/p/11689641.html

 

https://developer.android.com/ndk/guides

 

責任編輯:武曉燕 來源: 程序喵大人
相關推薦

2020-03-27 08:43:20

Vueprops開發

2018-01-29 15:23:14

網絡知識點軟件測試

2012-04-23 15:49:04

2021-02-26 22:34:28

Webpack 前端項目

2024-03-05 08:33:52

OptionsAPIcomuted

2021-08-11 08:16:02

springboot 動態注冊項目

2021-06-11 11:42:57

Swift 函數生成器結果生成器

2009-04-01 11:39:39

視圖DB2

2010-08-30 19:42:45

DHCP服務器

2010-08-17 14:56:00

HCNE認證

2011-04-15 12:25:21

BGP路由

2016-05-30 17:31:34

Spring框架

2022-08-03 08:03:03

前端APIjavascript

2019-12-03 08:13:06

BDRDR路由器

2021-06-29 15:56:39

MYSQL開發數據庫

2017-07-04 13:59:28

Android模塊化

2021-06-17 06:51:32

Java泛型Java編程

2017-12-08 14:26:19

Android面試知識點總結

2020-04-15 11:21:49

QQ騰訊

2022-01-12 14:24:37

接口Callable程序
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 最新日韩在线 | 午夜免费观看 | 国产成人av电影 | 中文字幕免费视频 | 五月天婷婷丁香 | 97国产精品 | 成人一区二区三区在线观看 | 青青久草 | 不卡一二三区 | 91久久精品国产免费一区 | 欧美涩涩网 | 国产伦精品一区二区三区照片91 | 台湾av在线| 中文一区 | aaa级片| 国产99视频精品免费播放照片 | 国产精品不卡视频 | 综合网在线 | 狠狠躁18三区二区一区 | 综合色久 | 欧美精品一区在线 | 久久国产欧美一区二区三区精品 | 最新中文字幕一区 | 在线观看免费观看在线91 | 国产成人精品久久二区二区91 | 亚洲成人一二三 | 久久久久国产精品一区二区 | 欧美成人a∨高清免费观看 老司机午夜性大片 | 日韩羞羞 | 国产免国产免费 | 国产色黄| 亚洲视频在线播放 | 中文天堂在线观看 | 一级做a爰片性色毛片 | 97超在线视频 | 精品国产欧美在线 | 美女黄视频网站 | 成人精品鲁一区一区二区 | 亚洲精品久久久久久一区二区 | av在线一区二区 | 国产一区二区三区久久久久久久久 |