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 中的枚举类及单例模式的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: JSONP简单入门介绍
- 下一篇: Angular开发者指南(五)服务