23JNI與Android VM之關系.doc_第1頁
23JNI與Android VM之關系.doc_第2頁
23JNI與Android VM之關系.doc_第3頁
23JNI與Android VM之關系.doc_第4頁
23JNI與Android VM之關系.doc_第5頁
全文預覽已結束

下載本文檔

版權說明:本文檔由用戶提供并上傳,收益歸屬內容提供方,若內容存在侵權,請進行舉報或認領

文檔簡介

23.JNI與AndroidVM之關系1.從如何載入*.so檔案談起由于Android的應用層級類別都是以Java撰寫的,這些Java類別轉譯為Dex型式的Bytecode之后,必須仰賴Dalvik虛擬機器(VM:VirtualMachine)來執行之。VM在Android平臺里,扮演很重要的角色。此外,在執行Java類別的過程中,如果Java類別需要與C組件溝通時,VM就會去加載C組件,然后讓Java的函數順利地呼叫到C組件的函數。此時,VM扮演著橋梁的角色,讓Java與C組件能透過標準的JNI接口而相互溝通。應用層級的Java類別是在虛擬機器(VM:VitualMachine)上執行的,而C組件不是在VM上執行,那么Java程序又如何要求VM去加載(Load)所指定的C組件呢?可使用下述指令:System.loadLibrary(*.so的檔名);例如,在上一節的范例里的NativeJniAdder類別,其程序代碼:看看Android框架里所提供的MediaPlayer.java類別,內含指令:publicclassMediaPlayerstaticSystem.loadLibrary(media_jni);.這要求VM去加載Android的/system/lib/libmedia_jni.so檔案。載入*.so檔之后,Java類別與*.so檔就匯合起來,一起執行了。2.如何撰寫*.so的入口函數-JNI_OnLoad()與JNI_OnUnload()函數之用途當VM執行到System.loadLibrary()函數時,首先會去執行C組件里的JNI_OnLoad()函數。它的用途有二:1.告訴VM此C組件使用那一個JNI版本。如果你的*.so文件沒有提供JNI_OnLoad()函數,VM會默認該*.so檔是使用最老的JNI1.1版本。由于新版的JNI做了許多擴充,如果需要使用JNI的新版功能,例如JNI1.4的java.nio.ByteBuffer,就必須藉由JNI_OnLoad()函數來告知VM。2.由于VM執行到System.loadLibrary()函數時,就會立即先呼叫JNI_OnLoad(),所以C組件的開發者可以藉由JNI_OnLoad()來進行C組件內的初期值之設定(Initialization)。例如,在Android的/system/lib/libmedia_jni.so檔案里,就提供了JNI_OnLoad()函數,其程序代碼片段為:/#defineLOG_NDEBUG0#defineLOG_TAGMediaPlayer-JNIjintJNI_OnLoad(JavaVM*vm,void*reserved)JNIEnv*env=NULL;jintresult=-1;if(vm-GetEnv(void*)&env,JNI_VERSION_1_4)!=JNI_OK)LOGE(ERROR:GetEnvfailedn);gotobail;assert(env!=NULL);if(register_android_media_MediaPlayer(env)0)LOGE(ERROR:MediaPlayernativeregistrationfailedn);gotobail;if(register_android_media_MediaRecorder(env)0)LOGE(ERROR:MediaRecordernativeregistrationfailedn);gotobail;if(register_android_media_MediaScanner(env)0)LOGE(ERROR:MediaScannernativeregistrationfailedn);gotobail;if(register_android_media_MediaMetadataRetriever(env)0)LOGE(ERROR:MediaMetadataRetrievernativeregistrationfailedn);gotobail;/*success-returnvalidversionnumber*/result=JNI_VERSION_1_4;bail:returnresult;/KTHXBYE此函數回傳JNI_VERSION_1_4值給VM,于是VM知道了其所使用的JNI版本了。此外,它也做了一些初期的動作(可呼叫任何本地函數),例如指令:if(register_android_media_MediaPlayer(env)GetEnv(void*)&env,JNI_VERSION_1_4)!=JNI_OK)LOGE(ERROR:GetEnvfailedn);gotobail;由于VM通常是多執行緒(Multi-threading)的執行環境。每一個執行緒在呼叫JNI_OnLoad()時,所傳遞進來的JNIEnv指標值都是不同的。為了配合這種多執行緒的環境,C組件開發者在撰寫本地函數時,可藉由JNIEnv指標值之不同而避免執行緒的數據沖突問題,才能確保所寫的本地函數能安全地在Android的多執行緒VM里安全地執行。基于這個理由,當在呼叫C組件的函數時,都會將JNIEnv指標值傳遞給它,如下:jintJNI_OnLoad(JavaVM*vm,void*reserved)JNIEnv*env=NULL;.if(register_android_media_MediaPlayer(env)MonitorEnter(env,obj)!=JNI_OK).查看是否已經有其它執行緒進入此對象,如果沒有,此執行緒就進入該對象里執行了。還有,也可撰寫下述指令:if(*env)-MonitorExit(env,obj)!=JNI_OK)查看是否此執行緒正在此對象內執行,如果是,此執行緒就會立即離開。3.registerNativeMethods()函數之用途應用層級的Java類別透過VM而呼叫到本地函數。一般是仰賴VM去尋找*.so里的本地函數。如果需要連續呼叫很多次,每次都需要尋找一遍,會多花許多時間。此時,組件開發者可以自行將本地函數向VM進行登記。例如,在Android的/system/lib/libmedia_jni.so檔案里的程序代碼片段如下:/#defineLOG_NDEBUG0#defineLOG_TAGMediaPlayer-JNIstaticJNINativeMethodgMethods=setDataSource,(Ljava/lang/String;)V,(void*)android_media_MediaPlayer_setDataSource,setDataSource,(Ljava/io/FileDescriptor;JJ)V,(void*)android_media_MediaPlayer_setDataSourceFD,prepare,()V,(void*)android_media_MediaPlayer_prepare,prepareAsync,()V,(void*)android_media_MediaPlayer_prepareAsync,_start,()V,(void*)android_media_MediaPlayer_start,_stop,()V,(void*)android_media_MediaPlayer_stop,getVideoWidth,()I,(void*)android_media_MediaPlayer_getVideoWidth,getVideoHeight,()I,(void*)android_media_MediaPlayer_getVideoHeight,seekTo,(I)V,(void*)android_media_MediaPlayer_seekTo,_pause,()V,(void*)android_media_MediaPlayer_pause,isPlaying,()Z,(void*)android_media_MediaPlayer_isPlaying,getCurrentPosition,()I,(void*)android_media_MediaPlayer_getCurrentPosition,getDuration,()I,(void*)android_media_MediaPlayer_getDuration,_release,()V,(void*)android_media_MediaPlayer_release,_reset,()V,(void*)android_media_MediaPlayer_reset,setAudioStreamType,(I)V,(void*)android_media_MediaPlayer_setAudioStreamType,setLooping,(Z)V,(void*)android_media_MediaPlayer_setLooping,setVolume,(FF)V,(void*)android_media_MediaPlayer_setVolume,getFrameAt,(I)Landroid/graphics/Bitmap;,(void*)android_media_MediaPlayer_getFrameAt,native_setup,(Ljava/lang/Object;)V,(void*)android_media_MediaPlayer_native_setup,native_finalize,()V,(void*)android_media_MediaPlayer_native_finalize,;staticintregister_android_media_MediaPlayer(JNIEnv*env)returnAndroidRuntime:registerNativeMethods(env,android/media/MediaPlayer,gMethods,NELEM(gMethods);./jintJNI_OnLoad(JavaVM*vm,void*reserved)if(register_android_media_MediaPlayer(env)0)LOGE(ERROR:MediaPlayernativeregistrationfailedn);gotobail;.當VM載入libmedia_jni.so檔案時,就呼叫JNI_OnLoad()函數。接著,JNI_OnLoad()呼叫register_android_media_MediaPlayer()

溫馨提示

  • 1. 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請下載最新的WinRAR軟件解壓。
  • 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請聯系上傳者。文件的所有權益歸上傳用戶所有。
  • 3. 本站RAR壓縮包中若帶圖紙,網頁內容里面會有圖紙預覽,若沒有圖紙預覽就沒有圖紙。
  • 4. 未經權益所有人同意不得將文件中的內容挪作商業或盈利用途。
  • 5. 人人文庫網僅提供信息存儲空間,僅對用戶上傳內容的表現方式做保護處理,對用戶上傳分享的文檔內容本身不做任何修改或編輯,并不能對任何下載內容負責。
  • 6. 下載文件中如有侵權或不適當內容,請與我們聯系,我們立即糾正。
  • 7. 本站不保證下載資源的準確性、安全性和完整性, 同時也不承擔用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。

最新文檔

評論

0/150

提交評論