在线观看www成人影院-在线观看www日本免费网站-在线观看www视频-在线观看操-欧美18在线-欧美1级

0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

Java反射機(jī)制清空字符串導(dǎo)致業(yè)務(wù)異常分析

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

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

背景知識(shí)

JVM為了提高性能和減少內(nèi)存開銷,在實(shí)例化字符串常量時(shí)進(jìn)行了優(yōu)化。JVM在Java堆上開辟了一個(gè)字符串常量池空間(StringTable),JVM通過ldc指令加載字符串常量時(shí)會(huì)調(diào)用 StringTable::intern 函數(shù)將字符串加入到字符串常量池中。

  • StringTable::intern函數(shù)代碼
    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 函數(shù)處理流程

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

  • 字符串的創(chuàng)建方式

    根據(jù)StringTable::intern函數(shù)處理流程,我們可以簡(jiǎn)單描繪如下6種常見的字符串的創(chuàng)建方式以及引用關(guān)系。

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


		

現(xiàn)象

某業(yè)務(wù)線使用fastjson實(shí)現(xiàn)Java對(duì)象序列化功能,低概率出現(xiàn)接口返回的JSON數(shù)據(jù)的某個(gè)屬性值和實(shí)際值不一致的現(xiàn)象。正確的屬性值應(yīng)該為"null",實(shí)際屬性值卻為"0000"。

原因分析

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

一般導(dǎo)致常量池被修改有兩種可能性:

  1. 第三方動(dòng)態(tài)庫引入的bug導(dǎo)致字符串常量池內(nèi)容被修改;
  2. 在業(yè)務(wù)代碼中通過Java反射機(jī)制主動(dòng)修改字符串常量池內(nèi)容;

業(yè)務(wù)方排查項(xiàng)目中使用到的第三方動(dòng)態(tài)庫,未發(fā)現(xiàn)可疑的動(dòng)態(tài)庫,排除第一種可能性。排查業(yè)務(wù)代碼中使用到Java反射的功能,發(fā)現(xiàn)清空密碼功能會(huì)使用到Java反射機(jī)制,并且將String類型密碼的value數(shù)組元素全部設(shè)置為'0'。

業(yè)務(wù)出現(xiàn)的現(xiàn)象可以簡(jiǎn)單通過代碼模擬

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

復(fù)現(xiàn)代碼

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());
}
}

復(fù)現(xiàn)代碼字符串引用關(guān)系如下圖所示。

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

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

輸出結(jié)果如下:

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

通過輸出結(jié)果我們可以發(fā)現(xiàn)在通過Java反射機(jī)制修改某一個(gè)字符串內(nèi)容后,所有指向原字符串的引用的內(nèi)容全部變成修改后的內(nèi)容。

總結(jié)

在保存業(yè)務(wù)敏感數(shù)據(jù)時(shí)避免使用String類型保存,建議使用byte[]或char[]數(shù)組保存,然后通過Java反射機(jī)制清空敏感數(shù)據(jù)。

后記

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

審核編輯 :李倩


聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • 函數(shù)
    +關(guān)注

    關(guān)注

    3

    文章

    4374

    瀏覽量

    64400
  • JVM
    JVM
    +關(guān)注

    關(guān)注

    0

    文章

    160

    瀏覽量

    12542
  • 數(shù)組
    +關(guān)注

    關(guān)注

    1

    文章

    419

    瀏覽量

    26417

原文標(biāo)題:Java反射機(jī)制清空字符串導(dǎo)致業(yè)務(wù)異常分析

文章出處:【微信號(hào):openEulercommunity,微信公眾號(hào):openEuler】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。

收藏 人收藏

    評(píng)論

    相關(guān)推薦
    熱點(diǎn)推薦

    字符串在數(shù)據(jù)庫中的存儲(chǔ)方式

    數(shù)據(jù)庫是現(xiàn)代信息技術(shù)中存儲(chǔ)和管理數(shù)據(jù)的核心組件。字符串作為最常見的數(shù)據(jù)類型之一,在數(shù)據(jù)庫中的存儲(chǔ)方式對(duì)其性能和可擴(kuò)展性有著重要影響。 數(shù)據(jù)類型 固定長度字符串 :如CHAR類型,它為每個(gè)字符串分配
    的頭像 發(fā)表于 01-07 15:41 ?749次閱讀

    字符串在編程中的應(yīng)用實(shí)例

    字符串在編程中有著廣泛的應(yīng)用,它們被用于表示文本數(shù)據(jù)、處理用戶輸入、構(gòu)建動(dòng)態(tài)內(nèi)容等。以下是一些字符串在編程中的應(yīng)用實(shí)例: 1. 用戶輸入與輸出 用戶輸入 :程序通常需要從用戶那里獲取輸入,這些輸入通
    的頭像 發(fā)表于 01-07 15:33 ?659次閱讀

    字符串字符數(shù)組的區(qū)別

    在編程語言中,字符串字符數(shù)組是兩種基本的數(shù)據(jù)結(jié)構(gòu),它們都用于存儲(chǔ)和處理文本數(shù)據(jù)。盡管它們?cè)诠δ苌嫌幸欢ǖ闹丿B,但在內(nèi)部表示、操作方式和使用場(chǎng)景上存在顯著差異。 1. 內(nèi)部表示 字符串 字符串
    的頭像 發(fā)表于 01-07 15:29 ?1044次閱讀

    字符串反轉(zhuǎn)的實(shí)現(xiàn)方式

    在編程中,字符串反轉(zhuǎn)是一個(gè)基礎(chǔ)而重要的操作,它涉及到將一個(gè)字符串中的字符順序顛倒過來。這個(gè)操作在多種編程語言中都有不同的實(shí)現(xiàn)方式,本文將探討幾種常見的字符串反轉(zhuǎn)方法。 1. 遞歸方法
    的頭像 發(fā)表于 01-07 15:27 ?760次閱讀

    字符串處理方法 字符串轉(zhuǎn)數(shù)字的實(shí)現(xiàn)

    在編程中,將字符串轉(zhuǎn)換為數(shù)字是一個(gè)常見的需求。不同的編程語言有不同的方法來實(shí)現(xiàn)這一功能。以下是一些常見編程語言中的字符串轉(zhuǎn)數(shù)字的實(shí)現(xiàn)方法: Python 在Python中,可以使用內(nèi)置的 int
    的頭像 發(fā)表于 01-07 15:26 ?809次閱讀

    字符串處理:4G模組軟件指南精要!

    最近一直有朋友咨詢我關(guān)于4G模組的字符串處理,今天我便把相關(guān)指南展示給大家。
    的頭像 發(fā)表于 11-17 09:57 ?497次閱讀
    <b class='flag-5'>字符串</b>處理:4G模組軟件指南精要!

    base64字符串轉(zhuǎn)換為二進(jìn)制文件

    Base64是一種編碼方法,用于將二進(jìn)制數(shù)據(jù)轉(zhuǎn)換為ASCII字符串。這種編碼通常用于在不支持二進(jìn)制數(shù)據(jù)的系統(tǒng)中傳輸數(shù)據(jù),例如電子郵件或網(wǎng)頁。將Base64字符串轉(zhuǎn)換為二進(jìn)制文件的過程相對(duì)簡(jiǎn)單,但需要
    的頭像 發(fā)表于 11-10 10:55 ?2726次閱讀

    MATLAB(5)--字符串處理

    字符串表示 在MATLAB中,字符串是用單引號(hào)括起來的字符序列,是把一個(gè)字符串當(dāng)做一個(gè)行向量,這個(gè)行向量中,每個(gè)元素對(duì)應(yīng)一個(gè)字符。 若
    發(fā)表于 09-06 10:22

    labview字符串數(shù)組轉(zhuǎn)化為數(shù)值數(shù)組

    在LabVIEW中,將字符串數(shù)組轉(zhuǎn)換為數(shù)值數(shù)組是一項(xiàng)常見的任務(wù),尤其是在處理數(shù)據(jù)采集、信號(hào)處理或用戶輸入時(shí)。 1. 理解LabVIEW的數(shù)據(jù)類型 在開始之前,了解LabVIEW中的數(shù)據(jù)類型是非
    的頭像 發(fā)表于 09-04 17:47 ?5144次閱讀

    labview字符串如何轉(zhuǎn)換為16進(jìn)制字符串

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

    labview中如何實(shí)現(xiàn)字符串換行

    1. 字符串換行的基本概念 在LabVIEW中,字符串換行通常指的是在字符串中插入換行符,使得字符串在顯示或輸出時(shí)能夠自動(dòng)換行。這在創(chuàng)建用戶界面或處理文本數(shù)據(jù)時(shí)非常有用。 2.
    的頭像 發(fā)表于 09-04 15:47 ?3631次閱讀

    labview中如何實(shí)現(xiàn)字符串選擇輸出

    在LabVIEW中實(shí)現(xiàn)字符串選擇輸出是一項(xiàng)常見的任務(wù),它涉及到字符串處理、條件判斷和用戶界面設(shè)計(jì)等多個(gè)方面。由于LabVIEW是一種圖形化編程語言,其編程方式與傳統(tǒng)的文本編程語言有所不同,因此實(shí)現(xiàn)
    的頭像 發(fā)表于 09-04 15:44 ?2090次閱讀

    labview中常用的字符串函數(shù)有哪些?

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

    labview字符串的四種表示各有什么特點(diǎn)

    。在LabVIEW中,字符串是一種基本的數(shù)據(jù)類型,用于表示文本信息。字符串在LabVIEW中有多種表示方式,每種方式都有其特定的應(yīng)用場(chǎng)景和特點(diǎn)。以下是對(duì)LabVIEW中四種字符串表示方式的分析
    的頭像 發(fā)表于 09-04 15:40 ?1304次閱讀

    字符串寫入數(shù)據(jù)庫報(bào)錯(cuò)誤的字符串

    在向表中姓名、性別、畢業(yè)院校中寫入文字時(shí)會(huì)報(bào)錯(cuò),錯(cuò)誤為:錯(cuò)誤的字符串值。輸入數(shù)字能正常寫入。錯(cuò)誤vi為DB TOOLS insert Data.vi
    發(fā)表于 06-21 19:12
    主站蜘蛛池模板: 欧美特级黄 | 香蕉成人国产精品免费看网站 | 国产精品一区二区三区四区五区 | 国产成+人+综合+亚洲欧美丁香花 | 男女激情做爰叫床声视频偷拍 | 加勒比精品久久一区二区三区 | 经典三级一区二区三区视频 | 1024你懂的国产日韩欧美 | 久久国产乱子伦精品免费看 | 黄色网址你懂的 | 激情爱爱的免费视频 | 日本免费的一级绿象 | 亚欧毛片基地国产毛片基地 | 69久久夜色精品国产69 | 好爽好深太大了再快一点 | 五月六月婷婷 | 2018国产一级天天弄 | 护士巨好爽好大乳 | 日日摸夜夜添免费毛片小说 | 亚洲1234区乱码 | 男校霸把男校草玩出水男男 | 国语对白一区二区三区 | 无遮挡很爽很污很黄很色的网站 | 色中色官网 | 奇米影视四色7777久久精品 | 国产自产视频在线观看香蕉 | 久久精品亚洲 | 456成人免费高清视频 | 国产一区二区三区波多野吉衣 | 黄色的视频网站在线观看 | 黄色三级免费网站 | 久久www免费人成_看 | 久久久久女人精品毛片九一 | 禁漫画羞羞动漫入口 | 国内精品久久久久影 | 男人操女人免费视频 | 夜色成人| 国产精品11页 | 久久精品免费观看视频 | 日韩免费精品一级毛片 | 美女又黄又免费的视频 |