電腦程式的思維邏輯 (26) - 剖析包裝類 (上)

来源:http://www.cnblogs.com/swiftma/archive/2016/07/14/5668379.html
-Advertisement-
Play Games

包裝類分為三節來介紹,本節主要介紹基本用法和共同點:與基本類型的相互轉換、裝箱/拆箱、重寫的Object方法、Comparable介面、與String的相互轉換、常用常量、Number、以及不可變性 ... ...


包裝類

Java有八種基本類型,每種基本類型都有一個對應的包裝類。

包裝類是什麼呢?它是一個類,內部有一個實例變數,保存對應的基本類型的值,這個類一般還有一些靜態方法、靜態變數和實例方法,以方便對數據進行操作。

Java中,基本類型和對應的包裝類如下表所示:

基本類型 包裝類
boolean
Boolean
byte
Byte
short
Short
int
Integer
long Long
float Float
double Double
char Character

包裝類也都很好記,除了Integer和Character外,其他類名稱與基本類型基本一樣,只是首字母大寫。

包裝類有什麼用呢?Java中很多代碼(比如後續文章介紹的集合類)只能操作對象,為了能操作基本類型,需要使用其對應的包裝類,另外,包裝類提供了很多有用的方法,可以方便對數據的操作。

包裝類的基本使用是比較簡單的,但我們不僅會介紹其基本用法,還會介紹一些平時用的相對較少的功能,同時剖析其實現代碼,內容比較多,我們會分三節來介紹,本節主要介紹各個包裝類的基本用法及其共同點,後兩節我們會進一步介紹高級功能,並剖析實現代碼。

讓我們逐步來介紹。

基本類型和包裝類

我們先來看各個基本類型和其包裝類是如何轉換的,我們直接看代碼:

Boolean

boolean b1 = false;
Boolean bObj = Boolean.valueOf(b1);
boolean b2 = bObj.booleanValue();

Byte

byte b1 = 123;
Byte byteObj = Byte.valueOf(b1);
byte b2 = byteObj.byteValue();

Short

short s1 = 12345;
Short sObj = Short.valueOf(s1);
short s2 = sObj.shortValue();

Integer

int i1 = 12345;
Integer iObj = Integer.valueOf(i1);
int i2 = iObj.intValue();

Long

long l1 = 12345;
Long lObj = Long.valueOf(l1);
long l2 = lObj.longValue();

Float

float f1 = 123.45f;
Float fObj = Float.valueOf(f1);
float f2 = fObj.floatValue();

Double

double d1 = 123.45;
Double dObj = Double.valueOf(d1);
double d2 = dObj.doubleValue(); 

Character

char c1 = 'A';
Character cObj = Character.valueOf(c1);
char c2 = cObj.charValue(); 

這些代碼結構是類似的,每種包裝類都有一個靜態方法valueOf(),接受基本類型,返回引用類型,也都有一個實例方法xxxValue()返回對應的基本類型。

將基本類型轉換為包裝類的過程,一般稱為"裝箱",而將包裝類型轉換為基本類型的過程,則稱為"拆箱"。裝箱/拆箱寫起來比較啰嗦,Java 1.5以後引入了自動裝箱和拆箱技術,可以直接將基本類型賦值給引用類型,反之亦可,如下代碼所示:

Integer a = 100;
int b = a;

自動裝箱/拆箱是Java編譯器提供的能力,背後,它會替換為調用對應的valueOf()/xxxValue(),比如說,上面的代碼會被Java編譯器替換為:

Integer a = Integer.valueOf(100);
int b = a.intValue();

每種包裝類也都有構造方法,可以通過new創建,比如說:

Integer a = new Integer(100);
Boolean b = new Boolean(true);
Double d = new Double(12.345);
Character c = new Character('馬');

那到底應該用靜態的valueOf方法,還是使用new呢?一般建議使用valueOf。new每次都會創建一個新對象,而除了Float和Double外的其他包裝類,都會緩存包裝類對象,減少需要創建對象的次數,節省空間,提升性能,後續我們會分析其具體代碼。

重寫Object方法

所有包裝類都重寫了Object類的如下方法:

boolean equals(Object obj)
int hashCode()
String toString()

我們逐個來看下。

equals

equals用於判斷當前對象和參數傳入的對象是否相同,Object類的預設實現是比較地址,對於兩個變數,只有這兩個變數指向同一個對象時,equals才返回true,它和比較運算符(==)的結果是一樣的。

但,equals應該反映的是對象間的邏輯相等關係,所以這個預設實現一般是不合適的,子類需要重寫該實現。所有包裝類都重寫了該實現,實際比較用的是其包裝的基本類型值,比如說,對於Long類,其equals方法代碼是:

public boolean equals(Object obj) {
    if (obj instanceof Long) {
        return value == ((Long)obj).longValue();
    }
    return false;
}

對於Float,其實現代碼為:

public boolean equals(Object obj) {
    return (obj instanceof Float)
           && (floatToIntBits(((Float)obj).value) == floatToIntBits(value));
}

Float有一個靜態方法floatToIntBits(),將float的二進位表示看做int。需要註意的是,只有兩個float的二進位表示完全一樣的時候,equals才會返回true。在第5節的時候,我們提到小數計算是不精確的,數學概念上運算結果一樣,但電腦運算結果可能不同,比如說,看下麵代碼:

Float f1 = 0.01f;
Float f2 = 0.1f*0.1f;
System.out.println(f1.equals(f2));
System.out.println(Float.floatToIntBits(f1));
System.out.println(Float.floatToIntBits(f2)); 

輸出為:

false
1008981770
1008981771

也就是,兩個浮點數不一樣,將二進位看做整數也不一樣,相差為1。

Double的equals方法與Float類似,它有一個靜態方法doubleToLongBits,將double的二進位表示看做long,然後再按long比較。

hashCode

hashCode返回一個對象的哈希值,哈希值是一個int類型的數,由對象中一般不變的屬性映射得來,用於快速對對象進行區分、分組等。一個對象的哈希值不能變,相同對象的哈希值必須一樣。不同對象的哈希值一般應不同,但這不是必須的,可以有不同對象但哈希值相同的情況。

比如說,對於一個班的學生對象,hashCode可以是學生的出生月日,出生日期是不變的,不同學生生日一般不同,分佈比較均勻,個別生日相同的也沒關係。

hashCode和equals方法聯繫密切,對兩個對象,如果equals方法返回true,則hashCode也必須一樣。反之不要求,equal返回false時,hashCode可以一樣,也可以不一樣,但應該儘量不一樣。hashCode的預設實現一般是將對象的記憶體地址轉換為整數,子類重寫equals時,也必須重寫hashCode。之所以有這個規定,是因為Java API中很多類依賴於這個行為,尤其是集合中的一些類。

包裝類都重寫了hashCode,根據包裝的基本類型值計算hashCode,對於Byte, Short, Integer, Character,hashCode就是其內部值,代碼為:

public int hashCode() {
    return (int)value;
}

對於Boolean,hashCode代碼為:

public int hashCode() {
    return value ? 1231 : 1237;
}

根據基類類型值返回了兩個不同的數,為什麼選這兩個值呢?它們是質數,即只能被1和自己整除的數,後續我們會講到,質數比較好,但質數很多,為什麼選這兩個呢,這個就不得而知了,大概是因為程式員對它們有特殊的偏好吧。

對於Long,hashCode代碼為:

public int hashCode() {
    return (int)(value ^ (value >>> 32));
}

是高32位與低32位進行位異或操作。

對於Float,hashCode代碼為:

public int hashCode() {
    return floatToIntBits(value);
}

與equals方法類似,將float的二進位表示看做了int。

對於Double,hashCode代碼為:

public int hashCode() {
    long bits = doubleToLongBits(value);
    return (int)(bits ^ (bits >>> 32));
}

與equals類似,將double的二進位表示看做long,然後再按long計算hashCode。

關於equals和hashCode,我們還會在後續的章節中碰到,併進行進一步說明。

toString

每個包裝類也都重寫了toString方法,返回對象的字元串表示,這個一般比較自然,我們就不贅述了。

Comparable

每個包裝類也都實現了Java API中的Comparable介面,Comparable介面代碼如下:

public interface Comparable<T> {
    public int compareTo(T o);
}

<T>是泛型語法,我們後續文章介紹,T表示比較的類型,由實現介面的類傳入。介面只有一個方法compareTo,當前對象與參數對象進行比較,在小於、等於、大於參數時,應分別返回-1,0,1。

各個包裝類的實現基本都是根據基本類型值進行比較,不再贅述。對於Boolean,false小於true。對於Float和Double,存在和equals一樣的問題,0.01和0.1*0.1相比的結果並不為0。

包裝類和String

除了toString方法外,包裝類還有一些其他與String相關的方法。

除了Character外,每個包裝類都有一個靜態的valueOf(String)方法,根據字元串表示返回包裝類對象,如:

Boolean b = Boolean.valueOf("true");
Float f = Float.valueOf("123.45f");

也都有一個靜態的parseXXX(String)方法,根據字元串表示返回基本類型值,如:

boolean b = Boolean.parseBoolean("true");
double d = Double.parseDouble("123.45");

都有一個靜態的toString()方法,根據基本類型值返回字元串表示,如:

System.out.println(Boolean.toString(true));
System.out.println(Double.toString(123.45));

輸出:

true
123.45 

對於整數類型,字元串表示除了預設的十進位外,還可以表示為其他進位,如二進位、八進位和十六進位,包裝類有靜態方法進行相互轉換,比如:

System.out.println(Integer.toBinaryString(12345)); //輸出2進位
System.out.println(Integer.toHexString(12345)); //輸出16進位
System.out.println(Integer.parseInt("3039", 16)); //按16進位解析

輸出為:

11000000111001
3039
12345

常用常量

包裝類中除了定義靜態方法和實例方法外,還定義了一些靜態變數。

Boolean類型:

public static final Boolean TRUE = new Boolean(true);
public static final Boolean FALSE = new Boolean(false);

所有數值類型都定義了MAX_VALUE和MIN_VALUE,表示能表示的最大/最小值,比如,對Integer:

public static final int   MIN_VALUE = 0x80000000;
public static final int   MAX_VALUE = 0x7fffffff;

Float和Double還定義了一些特殊數值,比如正無窮、負無窮、非數值,如Double類:

public static final double POSITIVE_INFINITY = 1.0 / 0.0;
public static final double NEGATIVE_INFINITY = -1.0 / 0.0;
public static final double NaN = 0.0d / 0.0;

Number

六種數值類型包裝類有一個共同的父類Number,Number是一個抽象類,它定義瞭如下方法:

byte byteValue()
short shortValue()                
int intValue()
long longValue()
float floatValue()
double doubleValue()

通過這些方法,包裝類實例可以返回任意的基本數值類型。

不可變性

包裝類都是不可變類,所謂不可變就是,實例對象一旦創建,就沒有辦法修改了。這是通過如下方式強制實現的:

  • 所有包裝類都聲明為了final,不能被繼承
  • 內部基本類型值是私有的,且聲明為了final
  • 沒有定義setter方法

為什麼要定義為不可變類呢?不可變使得程式可以更為簡單安全,因為不用操心數據被意外改寫的可能了,可以安全的共用數據,尤其是在多線程的環境下。關於線程,我們後續文章介紹。

小結

本節介紹了包裝類的基本用法,基本類型與包裝類的相互轉換、自動裝箱/拆箱、重寫的Object方法、Comparable介面、與String的相互轉換、常用常量、Number父類,以及包裝類的不可變性。從日常基本使用來說,除了Character外,其他類介紹的內容基本就夠用了。

但Integer和Long中有一些關於位操作的方法,我們還沒有介紹,Character中的大部分方法我們也都沒介紹,它們的一些實現原理我們也沒討論,讓我們在接下來的兩節中繼續探索。

----------------

未完待續,查看最新文章,敬請關註微信公眾號“老馬說編程”(掃描下方二維碼),從入門到高級,深入淺出,老馬和你一起探索Java編程及電腦技術的本質。用心寫作,原創文章,保留所有版權。

-----------

更多好評原創文章

電腦程式的思維邏輯 (1) - 數據和變數

電腦程式的思維邏輯 (5) - 小數計算為什麼會出錯?

電腦程式的思維邏輯 (6) - 如何從亂碼中恢復 (上)?

電腦程式的思維邏輯 (8) - char的真正含義

電腦程式的思維邏輯 (12) - 函數調用的基本原理

電腦程式的思維邏輯 (17) - 繼承實現的基本原理

電腦程式的思維邏輯 (18) - 為什麼說繼承是把雙刃劍

電腦程式的思維邏輯 (19) - 介面的本質

電腦程式的思維邏輯 (20) - 為什麼要有抽象類?

電腦程式的思維邏輯 (21) - 內部類的本質

電腦程式的思維邏輯 (23) - 枚舉的本質

電腦程式的思維邏輯 (24) - 異常 (上)

電腦程式的思維邏輯 (25) - 異常 (下)

 


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

-Advertisement-
Play Games
更多相關文章
  • wordpress 已經完全部署到Linux後,進行開始安裝的時候,資料庫信息都填入好了(前提是:鏈接信息輸入都正確) 然後點擊會報錯,說是鏈接資料庫失敗(資料庫是建在阿裡雲伺服器上的),但是具體不知道是什麼錯誤 解決方案: 此時應該把php-fpm的錯誤日誌給打開,看看具體是什麼原因導致的 之後再 ...
  • 在java的世界里,提供了匿名內部類語法糖,用於幫助大家簡化代碼,本文簡要從介面,抽象類以及常規類描述其常用模式。 ...
  • 類似其它的語言, Python 中的函數使用小括弧( () )調用。函數在調用之前必須先定義。如果函數中沒有 return 語句, 就會自動返回 None 對象。 Python 是通過引用調用的。 這意味著函數內對參數的改變會影響到原始對象。不過事實上只有可變對象會受此影響, 對不可變對象來說, 它 ...
  • 現在企業開發中,我們都會創建一個javaWeb工程,在eclipse中指的是新建一個dynamic web project,創建完工程之後,我們在IDE中大體看到如下的工程目錄: 我們主要關心的文件夾有src、WebContent、WEB-INF、test等。 src:存放我們寫的java文件,以包 ...
  • 首先web.xml配置 url-pattern 配置/即可,如果配置/*的話會出問提,連返回視圖jsp也會攔截,具體原因請看翻看源碼 問提:這樣配置的話,大家都會忽略一個問題,就是a/b/c可以訪問,但是a/b/c.do;a/b/c.mm;a/b/c.zz都可以訪問a/b/c的controller, ...
  • spring aop是面向切麵編程,使用了動態代理的技術,這樣可以使業務邏輯的代碼不摻入其他亂七八糟的代碼 可以在切麵上實現合法性校驗、許可權檢驗、日誌記錄。。。 spring aop 用的多的有兩種配置方法:註解配置和xml配置 註解配置更好,友好重構、錯誤也好檢查 在這裡只講註解配置 aop概念 ...
  • 1. 1 char array1[100]; 2 char* array2; 二者都可以當指針用,使用時有何區別? 1中字元數組,存儲於棧區,sizeof(array1)為100,aray1代表首元素首地址,不可以被賦值,array1[0]這樣可以被賦值,數組大小確定,難以擴容 2中存儲於字元常量區 ...
  • Struts2 總結 1.控制器採用的是 Filter 這個伺服器端組件,相比於 Servlet ,Servlet 能獲取到的資源,Filter 也同樣能獲取到,不同的是 Filter 生命周期初始化和 Servlet 有所區別。 2.Struts2 使用 Filter 和 Interceptors ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...