成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

一段代碼搞崩Java,坑都埋到胸了!

開發 后端
數字運算,是一門語言安身立命的根本。如果連1+1都變得不可信了,整個程序就會變得不可信。

 [[413342]]

本文轉載自微信公眾號「小姐姐味道」,作者小姐姐養的狗。轉載本文請聯系小姐姐味道公眾號。

數字運算,是一門語言安身立命的根本。如果連1+1都變得不可信了,整個程序就會變得不可信。

考慮到這樣一段代碼:

  1. Integer a = 1; 
  2. System.out.println(a); 
  3. Integer b = 2; 
  4. System.out.println( a.intValue() == b.intValue() ); 
  5. System.out.println(a.equals(b)); 

執行的結果,竟然是:

  1. -996 
  2. true 
  3. true 

這時候,你還敢繼續把代碼寫下去么?

為什么會這樣?

很簡單,我們使用反射改變了某些東西。

下面這段代碼,將會改變一些基本運算的執行邏輯,理所當然屬于埋坑的范疇之一。我們還是先看一下它的行為。

  1. public class StaticBlock { 
  2.     static { 
  3.         try { 
  4.             Class<?> cls = Integer.class.getDeclaredClasses()[0]; 
  5.             Field f = cls.getDeclaredField("cache"); 
  6.             f.setAccessible(true); 
  7.             Integer[] cache = ((Integer[]) f.get(cls)); 
  8.             for (int i = 0; i < cache.length; i++) { 
  9.                 cache[i] = -996; 
  10.             } 
  11.         } catch (Exception e) { 
  12.             e.printStackTrace(); 
  13.             //silence 
  14.         } 
  15.     } 

程序使用反射,修改了Integer中cache變量中的內容,使得里面的數字,變成了一個固定的值。我們這里用的是-996,意思是永遠沒有996。

你只要想方設法把這段代碼給觸發了,Java的Integer包裝類,就算是廢了。

我們能這么做,關鍵就在于cache變量上。

數字緩存

Java 中有 8 種基本類型,鑒于 Java 面向對象的特點,它們同樣有著對應的 8 個包裝類型,比如 int 和 Integer,包裝類型的值可以為 null,很多時候,它們都能夠相互賦值。

考慮到下面這段小小的代碼,它的運算就經歷了多次裝箱拆箱。

  1. public Integer cal() { 
  2.  Integer a = 1000; 
  3.  int b = a * 10; 
  4.  return b; 

我們從字節碼層面看一下。

  1. public java.lang.Integer read(); 
  2.     descriptor: ()Ljava/lang/Integer
  3.     flags: ACC_PUBLIC 
  4.     Code: 
  5.       stack=2, locals=3, args_size=1 
  6.          0: sipush        1000 
  7.          3: invokestatic  #2                  // Method java/lang/Integer.valueOf:(I)Ljava/lang/Integer
  8.          6: astore_1 
  9.          7: aload_1 
  10.          8: invokevirtual #3                  // Method java/lang/Integer.intValue:()I 
  11.         11: bipush        10 
  12.         13: imul 
  13.         14: istore_2 
  14.         15: iload_2 
  15.         16: invokestatic  #2                  // Method java/lang/Integer.valueOf:(I)Ljava/lang/Integer
  16.         19: areturn 

可以看到這么簡單的運算,竟然涉及了valueOf、intValue等方法多次,說明它的計算過程效率是比單純的數字運算要低效的。

其中valueOf方法,用來將普通數字包裝成Integer,我們跟蹤到它的方法。

  1. public static Integer valueOf(int i) { 
  2.     if (i >= IntegerCache.low && i <= IntegerCache.high) 
  3.         return IntegerCache.cache[i + (-IntegerCache.low)]; 
  4.     return new Integer(i); 

為了增加轉化的效率,Integer內部,竟然緩存了i和Integer的對應關系!這樣在下次用的時候,就能夠直接進行定位。cache變量,就是用來存放這些中間信息的地方。如果我們通過反射改變了它,Integer就會有不正常的行為!

更多

IntegerCache,緩存了 low 和 high 之間的 Integer 對象,可以通過 -XX:AutoBoxCacheMax 來修改上限。

  1. String integerCacheHighPropValue = 
  2.                 sun.misc.VM.getSavedProperty("java.lang.Integer.IntegerCache.high"); 

有意思的是,Long也有這樣的Cache,但它的上下限是固定的,和Byte、Short是一樣的。

  1. static final Long cache[] = new Long[-(-128) + 127 + 1];S 

Double和Float比較慘,只能直接new一個,做不了這種緩存。

綜合來看,Integer是比較特殊的。下面這段代碼,即使我們不做反射魔改,它的輸出依然是不確定的。

  1. Integer n1 = 123; 
  2. Integer n2 = 123; 
  3. Integer n3 = 128; 
  4. Integer n4 = 128; 
  5.  
  6. System.out.println(n1 == n2); 
  7. System.out.println(n3 == n4); 

這是因為,正常情況,它會輸出true,false;而當我們使用AutoBoxCacheMax增加了它的上限,它就會輸出true,true。果然對象之間相互比較,還是得用equals才相對靠譜一點啊。

End

看著這個齊胸小坑,我的感情真的是難以言表。這段代碼整體看來,如果進行了正常的review,還是很容易看出問題的,但凡是總有萬一。

如果這段代碼被放到線上,哪怕是某個呆萌的同學不小心練手的時候提交到了倉庫中,后果都是毀滅性的。這段代碼目的比較直白,但如果我們把cache數組的修改邏輯,改的復雜一點,在某個特定的條件下才會觸發某單個變量值的修改,那才是要命的。

畢竟連sonar都掃描不出來,而且jdk中這樣的私有變量,還有一籮筐等著我們去探索呢!

作者簡介:小姐姐味道 (xjjdog),一個不允許程序員走彎路的公眾號。聚焦基礎架構和Linux。十年架構,日百億流量,與你探討高并發世界,給你不一樣的味道。

 

責任編輯:武曉燕 來源: 小姐姐味道
相關推薦

2015-03-27 11:34:59

JavaJava編寫引發內存泄露

2020-02-07 08:00:29

代碼Java8Bug

2020-11-24 06:17:57

微信代碼移動應用

2022-06-21 12:27:12

JavaScript前端

2020-12-31 10:14:42

防注入代碼繞過

2018-06-19 08:02:00

統計程序微信

2014-07-08 09:21:10

死代碼創意歌曲

2020-01-03 16:06:38

代碼開發工具

2021-02-04 07:55:28

代碼離職互聯網

2023-03-07 14:18:24

推特馬斯克

2020-07-14 13:31:07

程序員代碼Java

2020-06-17 11:06:25

GitHub代碼開發者

2021-04-29 23:45:07

函數式接口可用性

2018-11-02 16:16:41

程序硬盤存儲

2020-04-03 10:14:57

內存蠕蟲代碼web安全

2020-12-16 07:42:03

JS基礎代碼

2020-10-26 10:11:45

Jupyter Not早起Python開發

2014-03-21 09:58:08

比特幣

2019-03-22 09:13:47

淘寶12306閑魚

2021-10-28 19:35:02

代碼main方法
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 亚洲免费观看 | 亚洲精选久久 | 99久久精品免费看国产四区 | 精品一区二区在线观看 | 日韩视频一区二区 | 久久综合久久久 | 亚洲精品日韩精品 | 亚洲国产成人精品女人 | 高清一区二区三区 | 黄色av一区 | 久久www免费人成看片高清 | 国产999精品久久久 午夜天堂精品久久久久 | 日韩在线视频一区二区三区 | 91精品国产一区二区三区 | 中文字幕成人免费视频 | 农夫在线精品视频免费观看 | 九色porny自拍视频 | 成人国产精品久久 | 亚洲精品v | 久久久影院 | 色av一区二区三区 | 午夜精品一区二区三区免费视频 | 久久蜜桃资源一区二区老牛 | 日本天天色 | 国产精品99久久久久久大便 | 99国内精品久久久久久久 | 欧美精品久久久久久久久久 | 国产网站在线免费观看 | 国产一区二区三区在线 | 亚洲男人天堂 | 亚洲国产福利视频 | 91精品国产91久久久久久最新 | 最新中文字幕在线 | 久久免费视频1 | 狠狠操狠狠干 | 一区二区三区欧美 | 超碰操| 欧美精品一区在线 | 成人国产精品久久久 | 在线观看国产视频 | 日本天天操 |