23JNI與Android VM之關(guān)系.doc_第1頁(yè)
23JNI與Android VM之關(guān)系.doc_第2頁(yè)
23JNI與Android VM之關(guān)系.doc_第3頁(yè)
23JNI與Android VM之關(guān)系.doc_第4頁(yè)
23JNI與Android VM之關(guān)系.doc_第5頁(yè)
全文預(yù)覽已結(jié)束

下載本文檔

版權(quán)說(shuō)明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請(qǐng)進(jìn)行舉報(bào)或認(rèn)領(lǐng)

文檔簡(jiǎn)介

23.JNI與AndroidVM之關(guān)系1.從如何載入*.so檔案談起由于Android的應(yīng)用層級(jí)類別都是以Java撰寫(xiě)的,這些Java類別轉(zhuǎn)譯為Dex型式的Bytecode之后,必須仰賴Dalvik虛擬機(jī)器(VM:VirtualMachine)來(lái)執(zhí)行之。VM在Android平臺(tái)里,扮演很重要的角色。此外,在執(zhí)行Java類別的過(guò)程中,如果Java類別需要與C組件溝通時(shí),VM就會(huì)去加載C組件,然后讓Java的函數(shù)順利地呼叫到C組件的函數(shù)。此時(shí),VM扮演著橋梁的角色,讓Java與C組件能透過(guò)標(biāo)準(zhǔn)的JNI接口而相互溝通。應(yīng)用層級(jí)的Java類別是在虛擬機(jī)器(VM:VitualMachine)上執(zhí)行的,而C組件不是在VM上執(zhí)行,那么Java程序又如何要求VM去加載(Load)所指定的C組件呢?可使用下述指令:System.loadLibrary(*.so的檔名);例如,在上一節(jié)的范例里的NativeJniAdder類別,其程序代碼:看看Android框架里所提供的MediaPlayer.java類別,內(nèi)含指令:publicclassMediaPlayerstaticSystem.loadLibrary(media_jni);.這要求VM去加載Android的/system/lib/libmedia_jni.so檔案。載入*.so檔之后,Java類別與*.so檔就匯合起來(lái),一起執(zhí)行了。2.如何撰寫(xiě)*.so的入口函數(shù)-JNI_OnLoad()與JNI_OnUnload()函數(shù)之用途當(dāng)VM執(zhí)行到System.loadLibrary()函數(shù)時(shí),首先會(huì)去執(zhí)行C組件里的JNI_OnLoad()函數(shù)。它的用途有二:1.告訴VM此C組件使用那一個(gè)JNI版本。如果你的*.so文件沒(méi)有提供JNI_OnLoad()函數(shù),VM會(huì)默認(rèn)該*.so檔是使用最老的JNI1.1版本。由于新版的JNI做了許多擴(kuò)充,如果需要使用JNI的新版功能,例如JNI1.4的java.nio.ByteBuffer,就必須藉由JNI_OnLoad()函數(shù)來(lái)告知VM。2.由于VM執(zhí)行到System.loadLibrary()函數(shù)時(shí),就會(huì)立即先呼叫JNI_OnLoad(),所以C組件的開(kāi)發(fā)者可以藉由JNI_OnLoad()來(lái)進(jìn)行C組件內(nèi)的初期值之設(shè)定(Initialization)。例如,在Android的/system/lib/libmedia_jni.so檔案里,就提供了JNI_OnLoad()函數(shù),其程序代碼片段為:/#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此函數(shù)回傳JNI_VERSION_1_4值給VM,于是VM知道了其所使用的JNI版本了。此外,它也做了一些初期的動(dòng)作(可呼叫任何本地函數(shù)),例如指令:if(register_android_media_MediaPlayer(env)GetEnv(void*)&env,JNI_VERSION_1_4)!=JNI_OK)LOGE(ERROR:GetEnvfailedn);gotobail;由于VM通常是多執(zhí)行緒(Multi-threading)的執(zhí)行環(huán)境。每一個(gè)執(zhí)行緒在呼叫JNI_OnLoad()時(shí),所傳遞進(jìn)來(lái)的JNIEnv指標(biāo)值都是不同的。為了配合這種多執(zhí)行緒的環(huán)境,C組件開(kāi)發(fā)者在撰寫(xiě)本地函數(shù)時(shí),可藉由JNIEnv指標(biāo)值之不同而避免執(zhí)行緒的數(shù)據(jù)沖突問(wèn)題,才能確保所寫(xiě)的本地函數(shù)能安全地在Android的多執(zhí)行緒VM里安全地執(zhí)行。基于這個(gè)理由,當(dāng)在呼叫C組件的函數(shù)時(shí),都會(huì)將JNIEnv指標(biāo)值傳遞給它,如下:jintJNI_OnLoad(JavaVM*vm,void*reserved)JNIEnv*env=NULL;.if(register_android_media_MediaPlayer(env)MonitorEnter(env,obj)!=JNI_OK).查看是否已經(jīng)有其它執(zhí)行緒進(jìn)入此對(duì)象,如果沒(méi)有,此執(zhí)行緒就進(jìn)入該對(duì)象里執(zhí)行了。還有,也可撰寫(xiě)下述指令:if(*env)-MonitorExit(env,obj)!=JNI_OK)查看是否此執(zhí)行緒正在此對(duì)象內(nèi)執(zhí)行,如果是,此執(zhí)行緒就會(huì)立即離開(kāi)。3.registerNativeMethods()函數(shù)之用途應(yīng)用層級(jí)的Java類別透過(guò)VM而呼叫到本地函數(shù)。一般是仰賴VM去尋找*.so里的本地函數(shù)。如果需要連續(xù)呼叫很多次,每次都需要尋找一遍,會(huì)多花許多時(shí)間。此時(shí),組件開(kāi)發(fā)者可以自行將本地函數(shù)向VM進(jìn)行登記。例如,在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;.當(dāng)VM載入libmedia_jni.so檔案時(shí),就呼叫JNI_OnLoad()函數(shù)。接著,JNI_OnLoad()呼叫register_android_media_MediaPlayer()

溫馨提示

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

評(píng)論

0/150

提交評(píng)論