【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
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...