java單例之enum實現方式(java enum 單例)

本文目錄一覽:

Java 中 Enum 如何繼承?

Java Enum是不能繼承的,以下是解釋:

枚舉類使用enum定義後在編譯後默認繼承了java.lang.Enum類,而不是普通的繼承Object類。enum聲明類繼承了Serializable和Comparable兩個接口。且採用enum聲明後,該類會被編譯器加上final聲明(同String),故該類是無法繼承的。枚舉類的內部定義的枚舉值就是該類的實例(且必須在第一行定義,當類初始化時,這些枚舉值會被實例化)。

Java 5新增的enum關鍵詞,可以定義枚舉類。該類是一個特殊的類,可以定義自己的field、方法、可以實現接口,也可以定義自己的構造器。

在《Effective Java》中,為什麼推薦使用enum來實現單例模式

單例模式實現方式有很多:在第一次使用的時候創建(構造函數中判斷是否已經有實例存在),在類加載的時候用靜態塊兒創建(靜態塊初始化),在應用啟動的時候創建。

在單線程中,基本大同小異,保證類的實例在整個應用中只有一個,都是沒問題的。

但是在多線程環境下,什麼時候創建這個實例是要考慮線程安全的。

枚舉類型最大的特點就是:構造函數是private修飾的,也就是不能對其進行new,對象的實例都是預定義的,也就是在類加載的時候都是定義好了的,不會給其它調用去創建實例的機會。

結論是,可以模擬這個思路去創建單例,不一定非得用枚舉,但是推薦用使用定義枚舉的方式去實現單例模式

為什麼java中用枚舉實現單例模式會更好

單例模式實現方式有很多:在第一次使用的時候創建(構造函數中判斷是否已經有實例存在),在類加載的時候用靜態塊兒創建(靜態塊初始化),在應用啟動的時候創建。

在單線程中,基本大同小異,保證類的實例在整個應用中只有一個,都是沒問題的。

但是在多線程環境下,什麼時候創建這個實例是要考慮線程安全的。

枚舉類型最大的特點就是:構造函數是private修飾的,也就是不能對其進行new,對象的實例都是預定義的,也就是在類加載的時候都是定義好了的,不會給其它調用去創建實例的機會。

結論是,可以模擬這個思路去創建單例,不一定非得用枚舉,但是推薦用使用定義枚舉的方式去實現單例模式

Java 枚舉型為什麼是靜態的,以及是怎麼實現的

是的,Java枚舉型是靜態常量,隱式地用static final修飾過。

確切地說,Java枚舉型是“靜態常量”,這裡面包含了兩層意思:

枚舉型中的實例隱式地用static final修飾過。

枚舉型作為某個類中的成員字段也隱式地用static final修飾過。

public class Traffic{

public enum Light{GREEN,YELLOW,RED}

}

還是你上面這個代碼,反編譯一下,你就能看到–編譯器背着你偷偷做了哪些手腳:

/*

* Decompiled with CFR 0_118.

*/

package com.ciaoshen.thinkinjava.chapter19;

public class Traffic {

public static final class Light

extends EnumLight {

public static final /* enum */ Light GREEN = new Light();

public static final /* enum */ Light YELLOW = new Light();

public static final /* enum */ Light RED = new Light();

private static final /* synthetic */ Light[] $VALUES;

public static Light[] values() {

return (Light[])$VALUES.clone();

}

public static Light valueOf(String string) {

return Enum.valueOf(Light.class, string);

}

private Light() {

super(string, n);

}

static {

$VALUES = new Light[]{GREEN, YELLOW, RED};

}

}

}

首先,枚舉型Light是個實實在在的類。繼承自基類EnumLight。然後在你不知情的情況下,偷偷加了static final修飾詞。

然後三個枚舉實例GREEN, YELLOW, RED也確確實實是Light的實例。然而前面也加上了static final。

然後構造器也被偷偷地閹割成private。這種實例控制手段,是不是在單例器模式里也見過?所以枚舉也是實現單例器的一種方法。

然後編譯器還偷偷合成了Light[]數組,一個values( )方法,一個valueOf( )方法。這個values( )在Enum的文檔里都找不到。

如果在Enum里定義一個相關方法,你還會看到一個匿名內部類:

public enum Light{

GREEN{public void show(){System.out.println(“Green”);}},

YELLOW{public void show(){System.out.println(“Yellow”);}},

RED{public void show(){System.out.println(“Red”);}};

}

反編譯的結果如下:

/*

* Decompiled with CFR 0_118.

*/

package com.ciaoshen.thinkinjava.chapter18;

import java.io.PrintStream;

public class Light

extends EnumLight {

public static final /* enum */ Light GREEN = new Light(“GREEN”, 0){

public void show() {

System.out.println(“Green”);

}

};

public static final /* enum */ Light YELLOW = new Light(“YELLOW”, 1){

public void show() {

System.out.println(“Yellow”);

}

};

public static final /* enum */ Light RED = new Light(“RED”, 2){

public void show() {

System.out.println(“Red”);

}

};

private static final /* synthetic */ Light[] $VALUES;

public static Light[] values() {

return (Light[])$VALUES.clone();

}

public static Light valueOf(String string) {

return Enum.valueOf(Light.class, string);

}

private Light() {

super(string, n);

}

static {

$VALUES = new Light[]{GREEN, YELLOW, RED};

}

}

總之,Java的Enum枚舉型整個就是一個大大的“語法糖”。明明是一個完整的類,但只向用戶暴露幾個靜態常量,隱藏掉大部分實現的細節。

原創文章,作者:J33MY,如若轉載,請註明出處:https://www.506064.com/zh-hant/n/127346.html

(0)
打賞 微信掃一掃 微信掃一掃 支付寶掃一掃 支付寶掃一掃
J33MY的頭像J33MY
上一篇 2024-10-03 23:15
下一篇 2024-10-03 23:15

相關推薦

  • java client.getacsresponse 編譯報錯解決方法

    java client.getacsresponse 編譯報錯是Java編程過程中常見的錯誤,常見的原因是代碼的語法錯誤、類庫依賴問題和編譯環境的配置問題。下面將從多個方面進行分析…

    編程 2025-04-29
  • Java JsonPath 效率優化指南

    本篇文章將深入探討Java JsonPath的效率問題,並提供一些優化方案。 一、JsonPath 簡介 JsonPath是一個可用於從JSON數據中獲取信息的庫。它提供了一種DS…

    編程 2025-04-29
  • Java Bean加載過程

    Java Bean加載過程涉及到類加載器、反射機制和Java虛擬機的執行過程。在本文中,將從這三個方面詳細闡述Java Bean加載的過程。 一、類加載器 類加載器是Java虛擬機…

    編程 2025-04-29
  • Java騰訊雲音視頻對接

    本文旨在從多個方面詳細闡述Java騰訊雲音視頻對接,提供完整的代碼示例。 一、騰訊雲音視頻介紹 騰訊雲音視頻服務(Cloud Tencent Real-Time Communica…

    編程 2025-04-29
  • Java Milvus SearchParam withoutFields用法介紹

    本文將詳細介紹Java Milvus SearchParam withoutFields的相關知識和用法。 一、什麼是Java Milvus SearchParam without…

    編程 2025-04-29
  • Java 8中某一周的周一

    Java 8是Java語言中的一個版本,於2014年3月18日發布。本文將從多個方面對Java 8中某一周的周一進行詳細的闡述。 一、數組處理 Java 8新特性之一是Stream…

    編程 2025-04-29
  • Java判斷字符串是否存在多個

    本文將從以下幾個方面詳細闡述如何使用Java判斷一個字符串中是否存在多個指定字符: 一、字符串遍歷 字符串是Java編程中非常重要的一種數據類型。要判斷字符串中是否存在多個指定字符…

    編程 2025-04-29
  • VSCode為什麼無法運行Java

    解答:VSCode無法運行Java是因為默認情況下,VSCode並沒有集成Java運行環境,需要手動添加Java運行環境或安裝相關插件才能實現Java代碼的編寫、調試和運行。 一、…

    編程 2025-04-29
  • Java任務下發回滾系統的設計與實現

    本文將介紹一個Java任務下發回滾系統的設計與實現。該系統可以用於執行複雜的任務,包括可回滾的任務,及時恢復任務失敗前的狀態。系統使用Java語言進行開發,可以支持多種類型的任務。…

    編程 2025-04-29
  • Java 8 Group By 會影響排序嗎?

    是的,Java 8中的Group By會對排序產生影響。本文將從多個方面探討Group By對排序的影響。 一、Group By的概述 Group By是SQL中的一種常見操作,它…

    編程 2025-04-29

發表回復

登錄後才能評論