0024 Java學習筆記-面向對象-包裝類、對象的比較、String常量池問題

来源:http://www.cnblogs.com/sonng/archive/2016/11/18/6076045.html
-Advertisement-
Play Games

包裝類 基本類型 包裝類 + byte Byte + short Short + int Integer + long Long + char Character + float Float + double Double + boolean Boolean 基本類型轉為包裝類型: + 自動裝箱Au ...


包裝類

  • 基本類型-->包裝類
    • byte-->Byte
    • short-->Short
    • int-->Integer
    • long-->Long
    • char-->Character
    • float-->Float
    • double-->Double
    • boolean-->Boolean
  • 基本類型轉為包裝類型:
    • 自動裝箱Autoboxing:將基本類型直接賦值給包裝類變數或者Object類變數
  • 包裝類型轉為包裝類型:
    • 自動拆箱AutoUnboxing:把包裝類對象直接賦值給基本類型變數
  • 示例:自動裝箱與自動拆箱
public class T1{  
    public static void main(String[] args) {  
        Integer inObj=13;     //自動裝箱
        Object boolObj=true;  //自動裝箱,基本類型賦值給Object類型變數
        int i=inObj;          //自動拆箱
        float f=inObj;        //自動拆箱,註意類型匹配
        System.out.println(i);
        System.out.println(f);
        if (boolObj instanceof Boolean) {
            boolean b=(Boolean)boolObj; //強制類型轉換+自動拆箱
            System.out.println(b);
        }
    }
}
  • 字元串型的值轉為基本類型:
    • 用包裝類的靜態的parse...()方法。註意沒有Character類,String的toCharArray()方法轉為字元數組
    • 用包裝類的構造器
  • 基本類型轉為字元串
    • 靜態方法String.valueOf();
    • 基本變數+"";
  • 示例:字元串轉基本類型
public class Test{  
    public static void main(String[] args) {  
        String strByte="127";
        String strShort="32767";
        String strInt="2147483647";
        String strLong="21474836478";     //加了L,用parse和new轉,都會失敗
        String strFloat="3.1415F";        //加了F,用parse和new轉,都能成功
        String strDouble="3.5665956565";
        String strBoolean="true";
        
        byte b1=Byte.parseByte(strByte);
        byte b2=new Byte(strByte);
        System.out.println(b1+"    "+b2);
        short s1=Short.parseShort(strShort);
        short s2=Short.parseShort(strShort);
        System.out.println(s1+"    "+s2);
        int i1=Integer.parseInt(strInt);
        int i2=new Integer(strInt);
        System.out.println(i1+"    "+i2);
        long l1=Long.parseLong(strLong);
        long l2=new Long(strLong);
        System.out.println(l1+"    "+l2);
        float f1=Float.parseFloat(strFloat);
        float f2=new Float(strFloat);
        System.out.println(f1+"    "+f2);
        double d1=Double.parseDouble(strDouble);
        double d2=new Double(strDouble);
        System.out.println(d1+"    "+d2);
        boolean boo1=Boolean.parseBoolean(strBoolean);
        boolean boo2=new Boolean(strBoolean);
        System.out.println(boo1+"    "+boo2);
        
    }
}

輸出

127 127
32767 32767
2147483647 2147483647
21474836478 21474836478
3.1415 3.1415
3.5665956565 3.5665956565
true true

  • 示例:基本類型轉字元串
public class Test{  
    public static void main(String[] args) {  
        System.out.println(String.valueOf(127));
        System.out.println(String.valueOf(32767));
        System.out.println(String.valueOf(2147483647));
        System.out.println(String.valueOf(21474836478L));  //註意L
        System.out.println(String.valueOf(3.1415F));       //註意F
        System.out.println(String.valueOf(3.5665956565));
        System.out.println(String.valueOf(true));
    }
}

輸出如下:

127
32767
2147483647
21474836478 //註意沒有L
3.1415 //註意沒有F
3.5665956565
true

  • 包裝類型與基本類型的比較
    • 包裝類是引用類型,但可以直接用"=="跟基本類型比較
  • 包裝類與包裝類的比較
    • 如果包裝類對象都是通過構造方法創建的:遵循對象的比較規則,即用"=="比較的是地址,用equals()比較對象的內容
    • 如果包裝類對象都是通過自動裝箱創建,且屬於[-128,127],可以用"=="比較他們的值是否相等;如果在[-128,127]之外,那麼遵循對象的比較規則
    • 通過自動裝箱和構造方法創建的對象,不論是不是屬於[-128,127],都遵循對象的比較規則
  • 關於[-128,127]:
    • Integer類在初始化時,一個內部類的靜態代碼塊將[-128,127]的每個整數都創建了Integer對象,並保存在一個數組中,今後在將[-128,127]中的一個整數自動裝箱成Integer對象時,則直接指向這個數組中對應的對象,因此可以用"=="比較它們的值是否相等
  • 示例:包裝類與包裝類的比較
public class Test{  
    public static void main(String[] args) {  
        System.out.println(new Integer(2)==new Integer(2));
        System.out.println(new Integer(2).equals(new Integer(2)));
        
        Integer i1=127;
        Integer i2=127;
        System.out.println("127: i1==i2?  "+(i1==i2));
        Integer i3=128;
        Integer i4=128;
        System.out.println("128: i3==i4?  "+(i3==i4));
        
        Integer i5=new Integer(127);
        System.out.println("127: i1==i5?  "+(i1==i5));
        Integer i6=new Integer(128);
        System.out.println("128: i3==i6?  "+(i3==i6));
    }
}

輸出如下:

false
true
127: i1==i2? true
128: i3==i4? false
127: i1==i5? false
128: i3==i6? false

  • 包裝類的compare()方法:compare(a,b)
    • a>b:返回1
    • a==b:返回0
    • a<b:返回-1
  • 示例代碼:
public class Test{  
    public static void main(String[] args) {  
        System.out.println(Boolean.compare(true,false));
        System.out.println(Boolean.compare(new Boolean(false),true));
        System.out.println(Integer.compare(5,13));
        System.out.println(Integer.compare(new Integer(19),new Integer(10)));
    }
}

toString()方法

對象與對象的相等比較equals()與"=="

  • "==":
    • 對引用變數而言,只有兩個變數指向同一個對象時,才返回true
    • 兩個對象如果沒有繼承關係,那麼不能用"=="比較,會出現編譯錯誤
  • equals():
    • Object的equals()方法跟"=="一樣,指向同一個對象才返回true
    • equals()一般都要重寫,重寫應滿足的條件:
      • 自反性:對任意x,x.equals(x)一定返回true
      • 對稱性:對任意x、y,如果x.equals(y)返回true,那麼y.equals(x)也要返回true
      • 傳遞性:對任意x、y、z,x.equals(y)返回true,y.equals(z)也返回true,那麼x.equals(z)一定也返回true
      • 一致性:只要兩個對象用於比較的信息沒有發生改變,那麼不論調用equals()多少次,返回的結構都應該相同
      • 對任意不是null的x,x.equals(null)一定返回false
  • equals()重寫示例,固定寫法
class Person{
    private String name;
    private String id;
    Person(){}
    Person(String name,String id){
        this.name=name;
        this.id=id;
    }
    public boolean equals(Object obj){
        if (this==obj){   //如果二者指向同一個對象,返回true
            return true;
        }
        if (obj!=null && obj.getClass()==Person.class){ //obj不為null,且指向的對象是Person類
            Person perObj=(Person)obj;          //類型轉換
            if(this.id.equals(perObj.id)){      //根據id是否相等判斷兩個Person對象是否相等
                return true;
            }
        }
        return false; //如果obj為null或者obj指向的對象不是Person,返回false
    }
}

字元串與"=="

  • 字元串有兩種情況,一種是字元串直接量,存儲在常量池中,常量池也在堆記憶體中;一種是堆記憶體中的字元串對象,存儲在堆記憶體中
  • String的這個問題很複雜,慢慢來看,下麵參考:http://www.cnblogs.com/kkgreen/archive/2011/08/24/2151450.html

代碼1:

String s1="AB";
String s2="A"+"B";
System.out.println(s1==s2);  //true。編譯階段即可確定s2="AB",運行時,s1和s2都指向常量池的"AB"

代碼2:

String s1="AB";
String s2="B";
String s3="A"+s2;
System.out.println(s1==s3);  //false。s2是變數,編譯階段不能確定s3的值

代碼3:

String s1="AB";
final String s2="B";   //註意多了個final
String s3="A"+s2;
System.out.println(s1==s3);  //true。s2的值不可變,編譯階段就確定了s3的值是"AB"

代碼4:

public static void main(String[] args) {  
        String s1="AB";
        final String s2=getString();  //註意有final
        String s3="A"+s2;
        System.out.println(s1==s3);   //false。雖然s2值不可變,但是是通過方法返回的,編譯階段也不能確定其值
    }
    static String getString(){
        return "B";
    }

代碼5:

String s1="AB";
String s2="A";
String s3="B";
String s4=s2+s3;
System.out.println(s1==s4); //false.

代碼:6

String s1="AB";
final String s2="A";
final String s3="B";
String s4=s2+s3;
System.out.println(s1==s4); //true.s4的值在編譯期即可確定

代碼7:開始引入intern()方法

String s1="A";
String s2="B";
String s3=s1+s2;                     //s3實際指向堆記憶體中的"AB"
System.out.println(s3==s3.intern()); //true。s3.intern()將堆記憶體中"AB"對象的地址添加到常量池表中,而不是在常量池中再創建個對象,二者實際都指向堆記憶體中的對象,因此二者相等
System.out.println(s3=="AB"); //true。s3指向堆記憶體中"AB"的地址;"AB"按理說應當位於常量池,但常量池中只保存了"AB"在堆記憶體中的地址,所有二者還是相當
//intern()方法用於將該字元串追加到常量池中,如果已經有了,就返回其引用;如果沒有就將地址添加到常量池表中,再返回其引用,實際指向的是堆記憶體中的對象

代碼8:將代碼7最後兩行代碼換個順序

String s1="A";
String s2="B";
String s3=s1+s2;
System.out.println(s3=="AB");         //false.s3指向堆記憶體中的對象,"AB"則在常量池中
System.out.println(s3==s3.intern());  //false.s3還是指向堆記憶體中的對象,s3.intern()返回常量池中的"AB"的地址
  • 總結:
    • 代碼7和8的結果十分詭異,以上所做的解釋也只是一種猜測,關鍵在於intern()方法,在常量池中沒有字元串的情況下,是新建個字元串對象,還是將字元串的地址添加在常量池表中
    • 凡是編譯階段能確定的字元串,在運行期就在常量池中創建一個運算後的對象,而不會再計算一遍,這點存疑,從Java1.7開始,常量池位於堆記憶體
    • 用""創建的字元串位於常量池
    • 用new String()構造方法創建的字元串位於堆記憶體中,運行期創建
    • 用"+"連接的字元串直接量,在編譯器就可以確定連接後的值,因此屬於常量池
    • 用"+"連接的是字元串和變數或者方法返回值,則要到運行期才能確定,屬於堆記憶體對象
  • 其他:
    • String s1=new String("ABC");String s2=new String("ABC");這兩個語句創建了3個String對象

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 在IIS中瀏覽某個網站時,出錯案例現場: 編譯器錯誤消息: CS0016: 未能寫入輸出文件“c:\Windows\Microsoft.NET\Framework64\v4.0.30319\Temporary ASP.NET Files\root\41c262f4\874fe77f\App_Web_ ...
  • 首先需要說明的是這是.net framework的一個組件,而不是針對.net core的。目前工作比較忙,因此.net core的轉換正在編寫過程中,有了實現會第一時間貼出來。 接下來進入正題。對於大型的分層系統,會有一個應用程式層,應用程式層的主要作用是封裝業務領域層的業務邏輯層,並對界面展示層 ...
  • 在上一篇文章中我們創建了WCF服務端應用程式,在這一篇文章中我們來學習如何創建WCF的服務端寄宿程式與客戶端調用程式。 ...
  • 環境準備 1、亞馬遜EC2 Windows Server 2016 with Container 2、Visual Studio 2015 Enterprise(Profresianal要裝Update 3) 3、.NET Core 1.0.0 – VS 2015 Tooling Preview 2 ...
  • .NET Core 1.1 RTM 版2016/11/16 發佈。對應發佈 ASP.NET Core 1.1 、EF Core 1.1。 你可以通過Visual Studio 2015, Visual Studio 2017 RC, Visual Studio Code and Visual Stu ...
  • 一. Xcode 環境安裝 與 工程創建 1. 下載環境 相關資源下載 : -- IOS 相關資料下載頁面 : https://developer.apple.com/devcenter/ios/index.action ; -- Xcode 下載頁面 : https://developer.app ...
  • 1. echo和print的區別 echo沒有返回值,print有返回值1,執行失敗時返回false;echo輸出的速度比print快,因為沒有返回值;echo可以輸出一個或多個字元串,print只允許輸出一個字元串;print可以用於複雜的表達式,echo不行。 另問:echo是個函數嗎? ech ...
  • 在PHP面試中,經常碰到此題 :要求寫出5種以上的方法,獲取一個文件的擴展名,其實也是在考察面試者基礎知識的掌握程度,下麵整理了幾種常用的方法(下麵方法返回的都是不帶’.'的,如果要求帶 ‘.’的話 自己改一下): <?php $file = ‘siyuantlw/程式設計.php’; functi ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...