精品国产人成在线_亚洲高清无码在线观看_国产在线视频国产永久2021_国产AV综合第一页一个的一区免费影院黑人_最近中文字幕MV高清在线视频

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

Java反射機制清空字符串導致業務異常分析

openEuler ? 來源:openEuler ? 作者:openEuler ? 2022-06-22 11:17 ? 次閱讀

編者按筆者在處理業務線問題時遇到接口返回的內容和實際內容不一致的現象。根因是業務方通過Java反射機制將String類型敏感數據引用的value數組元素全部設置為'0',從而實現清空用戶敏感數據的功能。這種清空用戶敏感數據的方法會將字符串常量池相應地址的內容修改,進而導致所有指向該地址的引用的內容和實際值不一致的現象。

背景知識

JVM為了提高性能和減少內存開銷,在實例化字符串常量時進行了優化。JVM在Java堆上開辟了一個字符串常量池空間(StringTable),JVM通過ldc指令加載字符串常量時會調用 StringTable::intern 函數將字符串加入到字符串常量池中。

  • StringTable::intern函數代碼
    oopStringTable::intern(Handlestring_or_null,jchar*name,
    intlen,TRAPS){
    unsignedinthashValue=hash_string(name,len);
    intindex=the_table()->hash_to_index(hashValue);
    oopfound_string=the_table()->lookup(index,name,len,hashValue);
    
    //Found
    if(found_string!=NULL){
    ensure_string_alive(found_string);
    returnfound_string;
    }
    
    debug_only(StableMemoryCheckersmc(name,len*sizeof(name[0])));
    assert(!Universe::heap()->is_in_reserved(name),
    "proposednameofsymbolmustbestable");
    
    Handlestring;
    //trytoreusethestringifpossible
    if(!string_or_null.is_null()){
    string=string_or_null;
    }else{
    string=java_lang_String::create_from_unicode(name,len,CHECK_NULL);
    }
    
    #ifINCLUDE_ALL_GCS
    if(G1StringDedup::is_enabled()){
    //Deduplicatethestringbeforeitisinterned.Notethatweshouldnever
    //deduplicateastringafterithasbeeninterned.Doingsowillcounteract
    //compileroptimizationsdoneone.g.internedstringliterals.
    G1StringDedup::deduplicate(string());
    }
    #endif
    
    //GrabtheStringTable_lockbeforegettingthe_table()becauseitcould
    //changeatsafepoint.
    oopadded_or_found;
    {
    MutexLockerml(StringTable_lock,THREAD);
    //Otherwise,addtosymboltotable
    added_or_found=the_table()->basic_add(index,string,name,len,
    hashValue,CHECK_NULL);
    }
    
    ensure_string_alive(added_or_found);
    
    returnadded_or_found;
    }
    
    
  • StringTable::intern 函數處理流程

    8346ea3e-f150-11ec-ba43-dac502259ad0.png

  • 字符串的創建方式

    根據StringTable::intern函數處理流程,我們可以簡單描繪如下6種常見的字符串的創建方式以及引用關系。

835414ac-f150-11ec-ba43-dac502259ad0.png


		

現象

某業務線使用fastjson實現Java對象序列化功能,低概率出現接口返回的JSON數據的某個屬性值和實際值不一致的現象。正確的屬性值應該為"null",實際屬性值卻為"0000"。

原因分析

為了排除fastjson自身的嫌疑,我們將其替換jackson后,依然會低概率出現同樣的現象。由于兩個不同三方件同時存在這個問題的可能性不大,為此我們暫時排除fastjson引入該問題的可能性。為了找到該問題的根因,我們在環境中開啟遠程調試功能。待問題復現,調試代碼時我們發現只要是指向"null"的引用,顯示的內容全部變成"0000",由此我們初步懷疑字符串常量池中的"null"被修改成"0000"。

一般導致常量池被修改有兩種可能性:

  1. 第三方動態庫引入的bug導致字符串常量池內容被修改;
  2. 在業務代碼中通過Java反射機制主動修改字符串常量池內容;

業務方排查項目中使用到的第三方動態庫,未發現可疑的動態庫,排除第一種可能性。排查業務代碼中使用到Java反射的功能,發現清空密碼功能會使用到Java反射機制,并且將String類型密碼的value數組元素全部設置為'0'。

業務出現的現象可以簡單通過代碼模擬

  1. 在TestString對象類中定義一個nullStr屬性,初始值為"null";
  2. 定義一個帶有password屬性的User類;
  3. 在main方法中創建一個密碼為"null"的User對象,使用Java反射機制將密碼字符串的所有字符全部修改為'0',分別在密碼修改前后打印TestString對象nullStr屬性值;

復現代碼

importjava.lang.reflect.Field;
importjava.util.Arrays;

publicclassTestString{
privateStringnullStr="null";

publicStringgetNullStr(){
returnnullStr;
}

staticclassUser{
privatefinalStringpassword;

User(Stringpassword){
this.password=password;
}

publicStringgetPassword(){
returnpassword;
}
}

privatestaticvoidclearPassword(Useruser)throwsException{
Fieldfield=String.class.getDeclaredField("value");
field.setAccessible(true);
char[]chars=(char[])field.get(user.getPassword());
Arrays.fill(chars,'0');
}

publicstaticvoidmain(String[]args)throwsException{
Useruser=newUser("null");
TestStringtestString=newTestString();
System.out.println("beforeclearpassword>>>>");
System.out.println("User.password:"+user.getPassword());
System.out.println("TestString.nullStr:"+testString.getNullStr());
System.out.println("--------------------------------");
clearPassword(user);
System.out.println("afterclearpassword>>>>");
System.out.println("User.password:"+user.getPassword());
System.out.println("TestString.nullStr:"+testString.getNullStr());
}
}

復現代碼字符串引用關系如下圖所示。

83671084-f150-11ec-ba43-dac502259ad0.png

User對象的password屬性和TestString的nullStr屬性引用都同時指向常量池中的"null"字符串,"null"字符串的value指向 {'n','u','l','l'} char數組。使用Java反射機制將User對象的password屬性引用的value數組全部設置為'0',導致TestString的nullStr屬性值也變成了 "0000"。

輸出結果如下:

beforeclearpassword>>>>
User.password:null
TestString.nullStr:null
--------------------------------
afterclearpassword>>>>
User.password:0000
TestString.nullStr:0000

通過輸出結果我們可以發現在通過Java反射機制修改某一個字符串內容后,所有指向原字符串的引用的內容全部變成修改后的內容。

總結

在保存業務敏感數據時避免使用String類型保存,建議使用byte[]或char[]數組保存,然后通過Java反射機制清空敏感數據。

后記

如果遇到相關技術問題(包括不限于畢昇 JDK),可以通過 Compiler SIG 求助。Compiler SIG 每雙周周二舉行技術例會,同時有一個技術交流群討論 GCC、LLVM 和 JDK 等相關編譯技術,感興趣的同學可以添加如下微信小助手入群。

審核編輯 :李倩


聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • 函數
    +關注

    關注

    3

    文章

    4308

    瀏覽量

    62445
  • JVM
    JVM
    +關注

    關注

    0

    文章

    157

    瀏覽量

    12212
  • 數組
    +關注

    關注

    1

    文章

    416

    瀏覽量

    25913

原文標題:Java反射機制清空字符串導致業務異常分析

文章出處:【微信號:openEulercommunity,微信公眾號:openEuler】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦

    labview字符串數組轉化為數值數組

    在LabVIEW中,將字符串數組轉換為數值數組是一項常見的任務,尤其是在處理數據采集、信號處理或用戶輸入時。 1. 理解LabVIEW的數據類型 在開始之前,了解LabVIEW中的數據類型是非
    的頭像 發表于 09-04 17:47 ?1841次閱讀

    labview字符串如何轉換為16進制字符串

    在LabVIEW中,將字符串轉換為16進制字符串是一個常見的需求,尤其是在處理數據通信和硬件接口時。LabVIEW提供了多種方法來實現這一轉換,包括使用內置函數、編寫VI(Virtual
    的頭像 發表于 09-04 15:54 ?1921次閱讀

    labview中常用的字符串函數有哪些?

    在LabVIEW中,常用的字符串函數廣泛覆蓋了對字符串的各種操作,包括但不限于格式化、搜索、替換、連接、計算長度等。以下是一些常用的字符串函數及其簡要說明: 字符串長度(String
    的頭像 發表于 09-04 15:43 ?542次閱讀

    labview字符串的四種表示各有什么特點

    。在LabVIEW中,字符串是一種基本的數據類型,用于表示文本信息。字符串在LabVIEW中有多種表示方式,每種方式都有其特定的應用場景和特點。以下是對LabVIEW中四種字符串表示方式的分析
    的頭像 發表于 09-04 15:40 ?438次閱讀

    C語言字符串編譯函數介紹

    在C語言中,字符串實際上是使用null字符O'終止的一維字符數組。因此,一個以null結尾的字符串,包含了組成字符串
    的頭像 發表于 03-07 16:18 ?481次閱讀
    C語言<b class='flag-5'>字符串</b>編譯函數介紹

    labview掃描字符串怎么用

    介紹如何在 LabVIEW 中使用掃描字符串以及相關的技巧和注意事項。 字符串是 LabVIEW 中的一種基本數據類型,表示一系列字符的序列。掃描字符串是指從一個給定的
    的頭像 發表于 12-29 10:12 ?1888次閱讀

    KUKA 機器人系統函數StrFind()-在字符串里查找

    搜索字符串變量 用功能 StrFind() 可以搜索字符串變量的字符串。 Result = StrFind( StartAt, StrVar[], StrFind[], CaseSens )
    的頭像 發表于 12-27 10:36 ?640次閱讀

    labview掃描字符串怎么用

    LabVIEW是一種圖形化編程語言,用于開發控制、測量和監控系統。雖然它主要用于工程和科學領域,但也可以用于處理文本和字符串。 在LabVIEW中,可以使用字符串處理函數來掃描字符串。以下是一些常用
    的頭像 發表于 12-26 16:58 ?1895次閱讀

    labview中怎么對字符串中的進行實時處理

    LabVIEW是一種用于開發控制、測試和測量系統的可視化編程環境,它提供了許多處理字符串的功能。在LabVIEW中,可以使用不同的函數和工具來實時處理字符串。下面我將詳細介紹一些常見的方法和技術
    的頭像 發表于 12-26 14:12 ?1797次閱讀

    oracle字符串split成多個

    Oracle是一種廣泛使用的關系型數據庫管理系統,它提供了許多強大的功能和函數,用于處理和操作數據。其中之一就是字符串分割(split)方法,該方法用于將一個字符串按照指定的分隔符分割成多個子字符串
    的頭像 發表于 12-06 09:54 ?5165次閱讀

    oracle判斷字符串包含某個字符

    字符串操作是任何編程語言中都非常重要的一部分,Oracle數據庫作為目前最常用的關系型數據庫之一,也提供了豐富的字符串操作函數和方法。在本文中,我們將詳細解析如何在Oracle中判斷一個字符串是否
    的頭像 發表于 12-06 09:53 ?1.5w次閱讀

    oracle中拼接字符串函數

    在Oracle中,我們可以使用 CONCAT 函數來拼接字符串。CONCAT 函數接受兩個參數,它將這兩個參數連接起來并返回相應的字符串結果。 語法示例: CONCAT(string1
    的頭像 發表于 12-06 09:49 ?2823次閱讀

    字符數組和字符串有沒有區別?

    字符數組和字符串有沒有區別?
    的頭像 發表于 11-30 16:39 ?630次閱讀

    java switch case 能不能用字符串

    Java的switch語句在處理字符串方面有一些限制。在Java 7之前,switch語句只能處理整數類型的表達式,如int或char。但從Java 7開始,
    的頭像 發表于 11-30 14:46 ?1975次閱讀

    MySQL替換字符串函數REPLACE

    MySQL是目前非常流行的開源數據庫管理系統之一,它具有強大的功能和性能。其中之一的字符串函數REPLACE,可以用于替換字符串中的指定字符字符串。在本文中,我們將詳細討論MySQL
    的頭像 發表于 11-30 10:44 ?1489次閱讀