日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

javese 5 中的枚举类及单例模式

發布時間:2025/3/20 编程问答 48 豆豆
生活随笔 收集整理的這篇文章主要介紹了 javese 5 中的枚举类及单例模式 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

為什么80%的碼農都做不了架構師?>>> ??

首先了解一下 javap 的用法,之后用到:javap -p (private)顯示所有類和成員,javap -c ?對代碼進行反匯編。

java.lang.Eumn<E>,在java代碼中使用關鍵字 enum 聲明一個枚舉類。你不能顯示地創建一個 Eumn 對象,因為這個類的構造器只允許編譯器調用。

通常實現一個枚舉,需要手動創建很多 static final 變量,就像下面這樣:

public static final int COLOR_RED = 0; public static final int COLOR_GREEN = 1; public static final int COLOR_YELLOW = 2;public void setColor(int color) {//some code here } //調用 setColor(COLOR_RED)

如果使用枚舉:

public enum Season {SPRING,SUMMER,AUTUMN,WINTER }

比上面那段代碼要簡潔很多,我們可以反編譯看看它的源碼:

E:\lemos\項目筆記\java基礎\javase\code\01>javap -p Season Compiled from "Season.java" public final class Season extends java.lang.Enum<Season> {public static final Season SPRING;public static final Season SUMMER;public static final Season AUTUMN;public static final Season WINTER;private static final Season[] $VALUES;public static Season[] values();public static Season valueOf(java.lang.String);private Season();static {}; }

可以看到這個類繼承自Enum,并設置為 final,也就是說這個類無法繼承其他類也不能被繼承。

具體編譯下,查看它的匯編碼,主要查看 static 代碼塊中包含哪些內容:

E:\lemos\項目筆記\java基礎\javase\code\01>javap -c Season Compiled from "Season.java" public final class Season extends java.lang.Enum<Season> {public static final Season SPRING;public static final Season SUMMER;public static final Season AUTUMN;public static final Season WINTER;public static Season[] values();Code:0: getstatic #1 // Field $VALUES:[LSeason;3: invokevirtual #2 // Method "[LSeason;".clone:()Ljava/lang/Object;6: checkcast #3 // class "[LSeason;"9: areturnpublic static Season valueOf(java.lang.String);Code:0: ldc #4 // class Season2: aload_03: invokestatic #5 // Method java/lang/Enum.valueOf:(Ljava/lang/Class;Ljava/lang/String;)Ljava/lang/Enum;6: checkcast #4 // class Season9: areturnstatic {};Code:0: new #4 // class Season3: dup4: ldc #7 // String SPRING6: iconst_07: invokespecial #8 // Method "<init>":(Ljava/lang/String;I)V10: putstatic #9 // Field SPRING:LSeason;13: new #4 // class Season16: dup17: ldc #10 // String SUMMER19: iconst_120: invokespecial #8 // Method "<init>":(Ljava/lang/String;I)V23: putstatic #11 // Field SUMMER:LSeason;26: new #4 // class Season29: dup30: ldc #12 // String AUTUMN32: iconst_233: invokespecial #8 // Method "<init>":(Ljava/lang/String;I)V36: putstatic #13 // Field AUTUMN:LSeason;39: new #4 // class Season42: dup43: ldc #14 // String WINTER45: iconst_346: invokespecial #8 // Method "<init>":(Ljava/lang/String;I)V49: putstatic #15 // Field WINTER:LSeason;52: iconst_453: anewarray #4 // class Season56: dup57: iconst_058: getstatic #9 // Field SPRING:LSeason;61: aastore62: dup63: iconst_164: getstatic #11 // Field SUMMER:LSeason;67: aastore68: dup69: iconst_270: getstatic #13 // Field AUTUMN:LSeason;73: aastore74: dup75: iconst_376: getstatic #15 // Field WINTER:LSeason;79: aastore80: putstatic #1 // Field $VALUES:[LSeason;83: return }

其中,

0~52為實例化SPRING, SUMMER, AUTUMN, WINTER
53~83為創建Season[]數組$VALUES,并將上面的四個對象放入數組的操作.

單例模式

  • 餓漢式加載
  • 懶漢式synchronize和雙重檢查
  • 利用java的靜態加載機制
  • 使用枚舉

餓漢式

public class Singleton{//類加載時就初始化private static final Singleton instance = new Singleton();private Singleton(){}public static Singleton getInstance(){return instance;} }

雙重校驗鎖( 使用雙重 if判斷,可以減少?synchronized 性能開銷)

這里使用 volatile 的目的是:避免重排序。直接原因也就是 初始化一個對象并使另一個引用指向他 這個過程可分為多個步驟:

  • 分配內存空間,
  • 初始化默認值(區別于構造器方法的初始化),
  • 初始化對象,
  • 將內存空間的地址賦值給對應的引用。

如果最后 2步替換順序,則導致了可能會出現引用指向了對象并未初始化好的那塊堆內存。

此外,注意 volatile 沒有解決操作的原子性,僅僅避免了重排序。即有可能出現一個線程執行了前2步,而另一線程重新執行的情況,所以需要加 synchronized 關鍵字保證此過程是單線程同步的。

public class Singleton {//volatile 防止延遲初始化private volatile static Singleton instance;public static Singleton getInstance() {if (instance == null) { //判斷是否已有單例生成synchronized (Singleton.class) {if (instance == null) //判斷是否已經賦值instance = new Singleton();//instance為volatile,現在沒問題了}}return instance;} }

由于類的靜態初始化會在類被加載時觸發,并且線程安全的。所以可以使用靜態內部類,同時實現懶加載。(靜態內部類是獨立的,不因基類加載而加載,只在被訪問時才加載,因此可以實現懶加載。并且因為靜態關系,內部類的任何靜態成員(包括要單例的對象)都會在其加載時被初始化)

public class SingleInstance {private SingleInstance() {}public static SingleInstance getInstance() {return SingleInstanceHolder.sInstance;}private static class SingleInstanceHolder {private static SingleInstance sInstance = new SingleInstance();} }

使用枚舉?

public enum AppManager {INSTANCE;private String tagName;public void setTag(String tagName) {this.tagName = tagName;}public String getTag() {return tagName;} }

當你調用時,只能通過 AppManager.INSTANCE 進行調用。比如:AppManager.INSTANCE.getTag(),這也滿足單例的兩個特性:按需加載和唯一實例。

轉載于:https://my.oschina.net/lemos/blog/842349

總結

以上是生活随笔為你收集整理的javese 5 中的枚举类及单例模式的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。