欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Java關鍵字之instanceof詳解

 更新時間:2021年11月04日 09:18:48   作者:YSOcean  
instanceof是Java的一個二元操作符,和==,>,<是同一類東東。由于它是由字母組成的,所以也是Java的保留關鍵字。它的作用是測試它左邊的對象是否是它右邊的類的實例,返回boolean類型的數據

instanceof 嚴格來說是Java中的一個雙目運算符,用來測試一個對象是否為一個類的實例,用法為:

boolean result = obj instanceof Class

其中 obj 為一個對象,Class 表示一個類或者一個接口,當 obj 為 Class 的對象,或者是其直接或間接子類,或者是其接口的實現類,結果result 都返回 true,否則返回false。

注意:編譯器會檢查 obj 是否能轉換成右邊的class類型,如果不能轉換則直接報錯,如果不能確定類型,則通過編譯,具體看運行時定。

1、obj 必須為引用類型,不能是基本類型

int i = 0;
System.out.println(i instanceof Integer);//編譯不通過
System.out.println(i instanceof Object);//編譯不通過

instanceof 運算符只能用作對象的判斷。

2、obj 為 null

System.out.println(null instanceof Object);//false

關于 null 類型的描述在官方文檔 有一些介紹。一般我們知道Java分為兩種數據類型,一種是基本數據類型,有八個分別是 byte short int long float double char boolean,一種是引用類型,包括類,接口,數組等等。而Java中還有一種特殊的 null 類型,該類型沒有名字,所以不可能聲明為 null 類型的變量或者轉換為 null 類型,null 引用是 null 類型表達式唯一可能的值,null 引用也可以轉換為任意引用類型。我們不需要對 null 類型有多深刻的了解,我們只需要知道 null 是可以成為任意引用類型的特殊符號。

在JavaSE規(guī)范中對 instanceof 運算符的規(guī)定就是:如果 obj 為 null,那么將返回 false。

3、obj 為 class 類的實例對象

Integer integer = new Integer(1);
System.out.println(integer instanceof  Integer);//true

這沒什么好說的,最普遍的一種用法。

4、obj 為 class 接口的實現類

了解Java 集合的,我們知道集合中有個上層接口 List,其有個典型實現類 ArrayList

public class ArrayList<E> extends AbstractList<E>
        implements List<E>, RandomAccess, Cloneable, java.io.Serializable

所以我們可以用 instanceof 運算符判斷 某個對象是否是 List 接口的實現類,如果是返回 true,否則返回 false

ArrayList arrayList = new ArrayList();
System.out.println(arrayList instanceof List);//true

或者反過來也是返回 true

List list = new ArrayList();
System.out.println(list instanceof ArrayList);//true

5、obj 為 class 類的直接或間接子類

我們新建一個父類 Person.class,然后在創(chuàng)建它的一個子類 Man.class

public class Person {
}

Man.class

public class Man extends Person{
}

測試:

Person p1 = new Person();
Person p2 = new Man();
Man m1 = new Man();
System.out.println(p1 instanceof Man);//false
System.out.println(p2 instanceof Man);//true
System.out.println(m1 instanceof Man);//true

注意第一種情況, p1 instanceof Man ,Man 是 Person 的子類,Person 不是 Man 的子類,所以返回結果為 false。

6、問題

前面我們說過編譯器會檢查 obj 是否能轉換成右邊的class類型,如果不能轉換則直接報錯,如果不能確定類型,則通過編譯,具體看運行時定。

看如下幾個例子:

Person p1 = new Person();
System.out.println(p1 instanceof String);//編譯報錯
System.out.println(p1 instanceof List);//false
System.out.println(p1 instanceof List<?>);//false
System.out.println(p1 instanceof List<Person>);//編譯報錯

按照我們上面的說法,這里就存在問題了,Person 的對象 p1 很明顯不能轉換為 String 對象,那么自然 Person 的對象 p1 instanceof String 不能通過編譯,但為什么 p1 instanceof List 卻能通過編譯呢?而 instanceof List<Person> 又不能通過編譯了?

7、深究原理

我們可以看Java語言規(guī)范Java SE 8 版:https://docs.oracle.com/javase/specs/jls/se8/html/jls-15.html#jls-15.20.2

在這里插入圖片描述

如果用偽代碼描述:

boolean result;
if (obj == null) {
  result = false;
} else {
  try {
      T temp = (T) obj; // checkcast
      result = true;
  } catch (ClassCastException e) {
      result = false;
  }
}

也就是說有表達式 obj instanceof T,instanceof 運算符的 obj 操作數的類型必須是引用類型或空類型; 否則,會發(fā)生編譯時錯誤。

如果 obj 強制轉換為 T 時發(fā)生編譯錯誤,則關系表達式的 instanceof 同樣會產生編譯時錯誤。 在這種情況下,表達式實例的結果永遠為false。

在運行時,如果 T 的值不為null,并且 obj 可以轉換為 T 而不引發(fā)ClassCastException,則instanceof運算符的結果為true。 否則結果是錯誤的

簡單來說就是:如果 obj 不為 null 并且 (T) obj 不拋 ClassCastException 異常則該表達式值為 true ,否則值為 false 。

所以對于上面提出的問題就很好理解了,為什么p1 instanceof String 編譯報錯,因為(String)p1 是不能通過編譯的,而 (List)p1 可以通過編譯。

8、instanceof 的實現策略

JavaSE 8 instanceof 的實現算法:https://docs.oracle.com/javase/specs/jvms/se8/html/jvms-6.html#jvms-6.5.instanceof

在這里插入圖片描述

1、obj如果為null,則返回false;否則設S為obj的類型對象,剩下的問題就是檢查S是否為T的子類型;

2、如果S == T,則返回true;

3、接下來分為3種情況,之所以要分情況是因為instanceof要做的是“子類型檢查”,而Java語言的類型系統(tǒng)里數組類型、接口類型與普通類類型三者的子類型規(guī)定都不一樣,必須分開來討論。

①、S是數組類型:如果 T 是一個類類型,那么T必須是Object;如果 T 是接口類型,那么 T 必須是由數組實現的接口之一;

②、接口類型:對接口類型的 instanceof 就直接遍歷S里記錄的它所實現的接口,看有沒有跟T一致的;

③、類類型:對類類型的 instanceof 則是遍歷S的super鏈(繼承鏈)一直到Object,看有沒有跟T一致的。遍歷類的super鏈意味著這個算法的性能會受類的繼承深度的影響。

參考鏈接:https://www.zhihu.com/question/21574535

相關文章

  • springboot讀取yml文件中的list列表、數組、map集合和對象方法實例

    springboot讀取yml文件中的list列表、數組、map集合和對象方法實例

    在平時的yml配置文件中,我們經常使用到配置基本數據類型的字符串,下面這篇文章主要給大家介紹了關于springboot讀取yml文件中的list列表、數組、map集合和對象的相關資料,需要的朋友可以參考下
    2023-02-02
  • Servlet連接數據庫實現用戶登錄的實現示例

    Servlet連接數據庫實現用戶登錄的實現示例

    本文主要介紹了Servlet連接數據庫實現用戶登錄的實現示例,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2022-06-06
  • Java 實現跨平臺的操作方式

    Java 實現跨平臺的操作方式

    這篇文章主要介紹了Java 實現跨平臺的操作方式,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2020-09-09
  • SpringBoot采用AJAX實現異步發(fā)布帖子詳解

    SpringBoot采用AJAX實現異步發(fā)布帖子詳解

    Ajax是一種web應用技術,可以借助客戶端腳本(javascript)與服務端應用進行異步通訊,獲取服務端數據以后,可以進行局部刷新,進而提高數據的響應和渲染速度。所有的Ajax請求都會基于DOM(HTML元素)事件,通過XHR(XMLHttpRequest)對象實現與服務端異步通訊局部更新
    2022-08-08
  • Java import static及import原理區(qū)別解析

    Java import static及import原理區(qū)別解析

    這篇文章主要介紹了Java import static及import原理區(qū)別解析,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下
    2020-10-10
  • Spring中@Autowired注解的原理詳解

    Spring中@Autowired注解的原理詳解

    這篇文章主要介紹了Spring中@Autowired注解的原理詳解,對于spring配置一個bean時,如果需要給該bean提供一些初始化參數,則需要通過依賴注入方式,所謂的依賴注入就是通過spring將bean所需要的一些參數傳遞到bean實例對象的過程,需要的朋友可以參考下
    2023-11-11
  • MyBatis的各種查詢功能結果接收類型的選擇(推薦)

    MyBatis的各種查詢功能結果接收類型的選擇(推薦)

    文章介紹了MyBatis中查詢結果的不同接收方式,包括單條數據和多條數據的處理方法,以及MyBatis的默認類型別名,感興趣的朋友跟隨小編一起看看吧
    2024-11-11
  • PowerJob的CleanService清理服務流程

    PowerJob的CleanService清理服務流程

    這篇文章主要為大家介紹了PowerJob的CleanService清理服務流程源碼解讀,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪<BR>
    2024-02-02
  • Springboot整合quartz實現多個定時任務實例

    Springboot整合quartz實現多個定時任務實例

    這篇文章主要介紹了Springboot整合quartz實現多個定時任務代碼實例,Quartz?是一款功能強大的開源任務調度框架,幾乎可以集成到任何?Java?應用程序中,Quartz?可用于創(chuàng)建簡單或復雜的任務調度,用以執(zhí)行數以萬計的任務,需要的朋友可以參考下
    2023-08-08
  • springmvc HttpServletRequest 如何獲取c:forEach的值

    springmvc HttpServletRequest 如何獲取c:forEach的值

    這篇文章主要介紹了springmvc HttpServletRequest 如何獲取c:forEach的值方法,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2021-08-08

最新評論