自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

Java迷題:等于,還是不等于?

開發(fā) 后端
表面上看來這只是一些基礎的問題,當真正的了解了以后你就會發(fā)現:啊,原來是這么一回事!下文是幾道Java謎題,不仔細分析就會犯錯哦。

等于還是不等于?

看來看下面的一段代碼:

代碼片段1

  1. public static void main(final String[] args) {    
  2.     Integer a = new Integer(100);    
  3.     Integer b = 100;    
  4.     System.out.println(a == b);     
  5. }   

這段代碼的輸出是什么?相信很多人都會很容易的猜到:false,因為a、b兩個對象的地址不同,用“==”比較時是false。恭喜你,答對了。

再看下面的一段代碼:

代碼片段2

  1. public static void main(final String[] args) {    
  2.     Integer a = 100;    
  3.     Integer b = 100;    
  4.     System.out.println(a == b);     
  5. }   

你可能會回答,這沒什么不一樣啊,所以還是false。很遺憾,如果你執(zhí)行上面的一段代碼,結果是true。

上面的代碼可能讓你有些意外,那好吧,再看看下面的這段代碼:

代碼片段3

  1. public static void main(final String[] args) {    
  2.     Integer a = 156;    
  3.     Integer b = 156;    
  4.     System.out.println(a == b);     
  5. }   

結果是true嗎?很遺憾,如果你執(zhí)行上面的一段代碼,結果是false。

感到吃驚嗎?那最后再看下面的一段代碼:

代碼片段4

  1. public static void main(final String[] args) {    
  2.     Integer a = Integer.valueOf(100);    
  3.     Integer b = 100;    
  4.     System.out.println(a == b);     
  5. }   

最后的結果,可能你已經猜到了,是true。

為什么會這樣?

現在我們分析一下上面的代碼。可以很容易的看出,這一系列代碼的最終目的都是用“==”對兩個對象進行比較。Java中,如果用“==”比較兩個對象結果為true,說明這兩個對象實際上是同一個對象,false說明是兩個對象。

現在,我們來看看為什么會出現上面的現象。

我們先看代碼片段4:最后的運行結果是true,說明a、b兩個對象實際上是同一個對象。但是a對象是通過調用Integer的valueOf方法創(chuàng)建的,而b對象是通過自動裝箱創(chuàng)建出來的,怎么會是同一個對象呢?難道問題在字節(jié)碼那里,畢竟Java程序是依靠虛擬器運行字節(jié)碼來實現的。

通過jdk中自帶的工具javap,解析字節(jié)碼,核心的部分摘取如下:

  1. 0:  bipush  100    
  2. 2:  invokestatic    #16//Method java/lang/Integer.valueOf:(I)Ljava/lang/Integer;    
  3. 5:  astore_1    
  4. 6:  bipush  100    
  5. 8:  invokestatic    #16//Method java/lang/Integer.valueOf:(I)Ljava/lang/Integer;   

代碼中我們只調用了一次Integer.valueOf方法,但是字節(jié)碼中出現了兩次對Integer.valueOf方法的調用。那么另一次是哪里呢?只可能在自動裝箱時調用的。因此這段代碼實際上等價于:

  1. public static void main(final String[] args) {    
  2.     Integer a = Integer.valueOf(100);    
  3.     Integer b = Integer.valueOf(100);    
  4.     System.out.println(a == b);     
  5. }   

現在問題就簡單了:看jdk源代碼,查看valueOf方法的具體實現:

  1. public static Integer valueOf(int i) {    
  2.     final int offset = 128;    
  3.     if (i >= -128 && i <= 127) { // must cache     
  4.         return IntegerCache.cache[i + offset];    
  5.     }    
  6.     return new Integer(i);    
  7. }   

看到這兒,上面的代碼就很明確了:對于-128到127的數字,valueOf返回的是緩存中的對象。所以兩次調用Integer.valueOf(100)返回的都是同一個對象。

我們再先看代碼片段3:根據上面的分析,代碼片段3實際上等價于以下代碼:

  1. public static void main(final String[] args) {    
  2.     Integer a = Integer.valueOf(156);    
  3.     Integer b = Integer.valueOf(156);    
  4.     System.out.println(a == b);     
  5. }   

由于156不在-128到127范圍內,所以兩個對象都是通過new Integer()的方式創(chuàng)建的,所以最后結果為false。

片段1和片段2就不做具體分析了,相信讀者可以自行分析。

最后,請大家思考一下問題:通過上面的分析,了解到整數的自動裝箱是通過Integer.valueOf(int number)實現的,那么自動拆箱是如何實現的呢?

原文鏈接:http://www.blogjava.net/todayx-org/archive/2012/02/01/369115.html

【編輯推薦】

  1. Java數據庫連接代碼集合
  2. Java之線程池簡單實現
  3. Java回調函數的理解
  4. Java路線圖:甲骨文的兩年計劃
  5. Java Thread的概述與總結
責任編輯:林師授 來源: todayx.org的博客
相關推薦

2015-08-12 10:04:24

2021-09-06 15:29:16

大數據防疫信息安全

2010-04-28 14:38:26

云計算

2017-06-29 08:45:06

MySQLNOT INNOT EXISTS

2010-10-18 10:51:00

蘋果

2010-07-19 11:12:43

Perl 不等于

2011-08-08 09:59:35

Android

2023-03-07 07:45:28

2023-06-02 13:53:56

2015-12-01 10:42:07

2013-11-26 09:55:12

2012-11-12 14:27:56

2009-03-11 11:36:31

技術人員角色感想

2012-11-14 09:54:14

2012-11-14 10:04:17

2013-08-15 09:15:23

云計算CIO信息化

2010-01-28 10:25:12

2012-04-17 09:27:21

工信部

2023-09-28 12:13:36

自然語言模型

2019-08-27 08:43:15

點贊
收藏

51CTO技術棧公眾號