【GiraKoo】Java Native Interface(JNI)的空間(引用)管理

来源:https://www.cnblogs.com/girakoo/archive/2023/05/18/17411752.html
-Advertisement-
Play Games

Java是通過垃圾回收機制回收記憶體,C/C++是通過malloc,free,new,delete手動管理空間。那麼在JNI層,同時存在Java和C/C++的空間時,該如何進行空間的管理呢?本文參考Oracle的官方文檔,對JNI層中空間的管理進行說明。明確哪些內容需要手動調用Delete,哪些不需要... ...


Java Native Interface(JNI)的空間(引用)管理

Java是通過垃圾回收機制回收記憶體,C/C++是通過malloc,free,new,delete手動管理空間。那麼在JNI層,同時存在Java和C/C++的空間時,該如何進行空間的管理呢?本文參考Oracle的官方文檔,對JNI層中空間的管理進行說明。明確哪些內容需要手動調用Delete,哪些不需要手動調用。

一、全局引用(Global References)

全局引用的生命周期(Lifetime),需要主動通過函數調用進行申請和釋放。native函數執行完畢後,該空間可繼續使用。

函數原型

// 創建全局引用
jobject NewGlobalRef(JNIEnv *env, jobject obj);  

// 刪除全局引用
void DeleteGlobalRef(JNIEnv *env, jobject globalRef);

二、局部引用(Local References)

局部引用的生命周期(Lifetime),與調用的native函數一致。當native函數return時,局部引用將會被自動釋放。該局部引用占用的空間是JVM的資源。

由於native函數用於存放局部引用的空間是固定的。如果過度的創建局部引用,不加以控制,可能會出現空間不足,程式拋出Out Of Memory(OOM)異常的問題。如果該局部引用已經使用完畢,應儘量手動調用DeleteLocalRef,提前釋放空間,避免OOM。

函數原型

// 創建全局引用
jobject NewLocalRef(JNIEnv *env, jobject obj);  

// 刪除全局引用
void DeleteLocalRef(JNIEnv *env, jobject localRef);

// 主動設置可創建的局部引用的數量
jint EnsureLocalCapacity(JNIEnv *env, jint capacity);

// 創建一個局部引用的幀,並指定可創建的局部引用的數量。
jint PushLocalFrame(JNIEnv *env, jint capacity);

// 釋放當前局部引用的幀,釋放全部本地引用。
// 可以通過參數result,將被PopLocalFrame釋放的引用轉移到上一層的局部引用幀中
jobject PopLocalFrame(JNIEnv *env, jobject result);

特殊說明

  1. 在JDK/JRE 1.1版本中,提供了DeleteLocalRef函數。
  2. 在JDK/JRE 1.2版本中,提供了EnsureLocalCapacityPushLocalFramePopLocalFrameNewLocalRef函數,支持更加複雜的局部引用生命周期能力。

三、弱全局引用(Weak Global Reference)

區別於全局引用(Global References),該引用並不會增加該引用的引用計數。

即持有弱全局引用,不會幹預垃圾回收機制(GC)對於該記憶體的回收。適用於生命周期短的對象,持有生命周期長的引用。

全局引用可以通過jboolean IsSameObject(JNIEnv *env, jobject ref1, jobject ref2); 函數,參數為被判定對象和NULL來判斷對象是否存在。如果返回JNI_TRUE判定該弱引用指向的對象是否已經被釋放。

使用弱全局引用時,可能會出現執行一半,引用被GC回收的情況。所以需要在使用弱全局引用之前,提供NewLocalRef或者NewGlobalRef,將弱引用轉換為強引用。確保GC不會對該空間進行回收。

四、常見問題

1、在native函數中,直接return一個jstring或者其他jobject,是否會有記憶體泄漏問題?

如果直接return的對象是局部引用(Local Reference),則不會發生記憶體泄漏問題。
局部引用在native函數執行完畢後,會根據局部引用表(Local Reference Table)進行空間的釋放。
當然,手動執行DeleteLocalRef函數進行提前釋放,也不會產生問題。

2、什麼情況下可能會出現記憶體泄漏?

1)創建的全局引用(Global Reference),未調用Delete函數進行回收。導致該對象引用計數無法歸零。
2)兩個對象互相持有對方的強引用。例如A持有B引用,B和C互相持有對方的引用。當A釋放B引用時,GC發現C依然持有B的引用,則不會釋放B。與此同時,GC也發現B持有C的引用,則不會釋放C。導致B和C與其他引用脫離,形成孤島。這塊記憶體將無法使用,也無法被釋放。

3、調用NewString和NewStringUTF時,是否需要釋放?

NewStringNewStringUTF函數創建的空間屬於局部引用(Local Reference)。可以等待native函數執行完畢後自動回收,也可以在使用完畢(後續不會再訪問)後,直接進行Delete釋放。

但是需要註意,如果通過GetStringCharsGetStringCritical或者GetStringUTFChars獲取了jstring中的char* 空間,則需要使用對應的ReleaseStringCharsReleaseStringCritical或者ReleaseStringUTFChars釋放該char* 空間。

參考文章

Logo
本文來自博客園,作者:GiraKoo
轉載請註明原文鏈接:https://www.cnblogs.com/girakoo/p/17411752.html
友情鏈接:GiraKoo 博客園 CSDN 稀土掘金


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

-Advertisement-
Play Games
更多相關文章
  • JWT介紹 JWT是JSON Web Token的縮寫,即JSON Web令牌,是一種自包含令牌。 是為了在網路應用環境間傳遞聲明而執行的一種基於JSON的開放標準。 JWT的聲明一般被用來在身份提供者和服務提供者間傳遞被認證的用戶身份信息,以便於從資源伺服器獲取資源。比如用在用戶登錄上。 JWT最 ...
  • java常用類 Object類 基類,超類,所有類的直接或間接父類 object類定義的方法是所有對象都具有的方法 object類型可以存儲任何對象 作為參數,可以接受任何對象 作為返回值,可以返回任何對象 getClass() 返回引用中存儲的實際對象類型 public class Student ...
  • Nacos 是一個開源的服務發現、配置管理和服務治理平臺,是阿裡巴巴開源的一款產品。 Nacos 可以幫助開發者更好地管理微服務架構中的服務註冊、配置和發現等問題,提高系統的可靠性和可維護性。 ...
  • Python字元串替換筆記主要展示瞭如何在Python中替換字元串。Python中有以下幾種替換字元串的方法,本文主要介紹前三種。 replace方法(常用) translate方法 re.sub方法 字元串切片(根據Python字元串切片方法替換字元) 1.replace方法 Python rep ...
  • 本文將演示在本地的 python 項目中調用 ChatGPT 模型 前言 作為一名程式員,在開發過程當中時常需要使用 ChatGPT 來完成一些任務,但總是使用網頁交互模式去 Web 端訪問 ChatGPT 是很麻煩的,這時候我們可以使用代碼來調用 ChatGPT 模型,以實現在本地和 Web 端一 ...
  • 摘要:這篇文章詳細介紹了頂帽運算和底帽運算,它們將為後續的圖像分割和圖像識別提供有效支撐。 本文分享自華為雲社區《[Python從零到壹] 四十九.圖像增強及運算篇之頂帽運算和底帽運算》,作者:eastmount。 數學形態學(Mathematical Morphology)是一種應用於圖像處理和模 ...
  • 基於java的校園二手交易系統或跳蚤市場設計與實現,java二手交易平臺,二手商城,交易商城,大學生交易平臺,springboot二手交易系統,二手交易平臺,購物平臺,大學生跳蚤平臺設計與實現,閑置物品交易平臺,校園閑置二手。 ...
  • Lambda 表達式(lambda expression)是一個匿名函數,Lambda表達式基於數學中的λ演算得名,直接對應於其中的lambda抽象(lambda abstraction),是一個匿名函數,即沒有函數名的函數。Lambda表達式可以表示閉包,和傳統數學上的意義有區別。 文末有本文重點 ...
一周排行
    -Advertisement-
    Play Games
  • .Net8.0 Blazor Hybird 桌面端 (WPF/Winform) 實測可以完整運行在 win7sp1/win10/win11. 如果用其他工具打包,還可以運行在mac/linux下, 傳送門BlazorHybrid 發佈為無依賴包方式 安裝 WebView2Runtime 1.57 M ...
  • 目錄前言PostgreSql安裝測試額外Nuget安裝Person.cs模擬運行Navicate連postgresql解決方案Garnet為什麼要選擇Garnet而不是RedisRedis不再開源Windows版的Redis是由微軟維護的Windows Redis版本老舊,後續可能不再更新Garne ...
  • C#TMS系統代碼-聯表報表學習 領導被裁了之後很快就有人上任了,幾乎是無縫銜接,很難讓我不想到這早就決定好了。我的職責沒有任何變化。感受下來這個系統封裝程度很高,我只要會調用方法就行。這個系統交付之後不會有太多問題,更多應該是做小需求,有大的開發任務應該也是第二期的事,嗯?怎麼感覺我變成運維了?而 ...
  • 我在隨筆《EAV模型(實體-屬性-值)的設計和低代碼的處理方案(1)》中介紹了一些基本的EAV模型設計知識和基於Winform場景下低代碼(或者說無代碼)的一些實現思路,在本篇隨筆中,我們來分析一下這種針對通用業務,且只需定義就能構建業務模塊存儲和界面的解決方案,其中的數據查詢處理的操作。 ...
  • 對某個遠程伺服器啟用和設置NTP服務(Windows系統) 打開註冊表 HKEY_LOCAL_MACHINE\SYSTEM\CurrentControlSet\Services\W32Time\TimeProviders\NtpServer 將 Enabled 的值設置為 1,這將啟用NTP伺服器功 ...
  • title: Django信號與擴展:深入理解與實踐 date: 2024/5/15 22:40:52 updated: 2024/5/15 22:40:52 categories: 後端開發 tags: Django 信號 松耦合 觀察者 擴展 安全 性能 第一部分:Django信號基礎 Djan ...
  • 使用xadmin2遇到的問題&解決 環境配置: 使用的模塊版本: 關聯的包 Django 3.2.15 mysqlclient 2.2.4 xadmin 2.0.1 django-crispy-forms >= 1.6.0 django-import-export >= 0.5.1 django-r ...
  • 今天我打算整點兒不一樣的內容,通過之前學習的TransformerMap和LazyMap鏈,想搞點不一樣的,所以我關註了另外一條鏈DefaultedMap鏈,主要調用鏈為: 調用鏈詳細描述: ObjectInputStream.readObject() DefaultedMap.readObject ...
  • 後端應用級開發者該如何擁抱 AI GC?就是在這樣的一個大的浪潮下,我們的傳統的應用級開發者。我們該如何選擇職業或者是如何去快速轉型,跟上這樣的一個行業的一個浪潮? 0 AI金字塔模型 越往上它的整個難度就是職業機會也好,或者說是整個的這個運作也好,它的難度會越大,然後越往下機會就會越多,所以這是一 ...
  • @Autowired是Spring框架提供的註解,@Resource是Java EE 5規範提供的註解。 @Autowired預設按照類型自動裝配,而@Resource預設按照名稱自動裝配。 @Autowired支持@Qualifier註解來指定裝配哪一個具有相同類型的bean,而@Resourc... ...