淺析Golang中怎麼實現列舉(附程式碼)

2022-02-04 10:00:17
Golang中怎麼實現列舉?下面本篇文章給大家介紹一下Golang中實現列舉的方法,希望對大家有所幫助!

在程式設計領域裡,列舉用來表示只包含有限數量的固定值的型別,在開發中一般用於標識錯誤碼或者狀態機。拿一個實體物件的狀態機來說,它通常與這個物件在資料庫裡對應記錄的標識狀態的欄位值相對應。【相關推薦:Go視訊教學

在剛開始學程式設計的時候,你一定寫過,至少見過直接使用魔術數位進行判斷的程式碼。啥叫魔術數位呢,舉個例子,要置頂一個文章的時候先判斷文章是不是已釋出狀態。

if (article.state == 2) {
   // state 2 代表文章已釋出
}

假如我們的程式碼裡沒有註釋,或者等我們專案的程式碼裡充斥著這些魔術數位的判斷的時候,你是不是會很頭疼?

後來我就學會了把這些狀態值定義成常數,並且也搞一個判斷物件狀態的方法單獨封裝這段邏輯。

public class ArticleState {
    
    public static final int Draft = 1; //草稿
    
    public static final int Published = 2; //釋出
    
    public static final int Deleted = 3; // 已刪除
}

public  Boolean checkArticleState(int state) {
    
    ...
    
}

這種用法,肯定是比在程式裡直接用魔術數位進行判斷要強很多啦,至少看著不會很頭疼,不會想罵**。

不過後來被當時帶我的老大哥說這種也有缺點,上面這個 checkArticleState 方法用來檢查文章狀態,本意是讓呼叫者傳入 ArticleState 的三個靜態常數之一,但由於沒有型別上的約束,因此傳入任意一個 int 值在語法上也是允許的,編譯器也不會提出任何警告,用列舉更合適一些。

em~! 我不記得大學教 Java 的那個學期老師講過這玩意啊,莫非又是一個上課玩手機錯過的知識點?所以使用列舉後我們的Java程式碼變成了:

// 使用enum而非class宣告
public enum ArticleState {
	
    //要在enum裡建立所有的列舉物件
    Draft(1, "草稿");
    Published(2, "已釋出");
    Deleted(3, "已刪除")
      
    // 自定義屬性
    private int code;

    private String text;
  
    // 構造方法必須是private的
    ArticleState(int code, String text) {
        this.code = id;
        this.text = name;
    }
}

public  Boolean checkArticleState(ArticleState state) {
    
    ...
    
}

這樣就能靠形參的列舉型別幫我們過濾掉非法的狀態值,把整型值作為引數傳給 checkArticleState 方法時因為型別不匹配編譯不過去,在寫程式碼是編譯器也能馬上提示出來。

如果沒有用過 Java 的小夥伴也不用糾結,主要語法點我用註釋標註出來了,大家應該都能看懂。

後來這兩年主要在用Go做專案,我發現相似的問題 Go 裡存在,但是 Go 並沒有提供列舉型別,那怎麼做到進行狀態值的正確限制呢?如果還是用 int 型的常數肯定不行。比如:

 const (
     Draft int = 1
     Published = 2
     Deleted   = 3
 )

 const (
     Summer int = 1
     Autumn     = 2
     Winter     = 3
     Spring     = 4
 )

 func main() {
     // 輸出 true, 不會有任何編譯錯誤
     fmt.Println(Autumn == Draft)
 }

比如上面定義了兩組 int 型別的常數,一類代表文章狀態,一類代表季節的四季。這種方式拿文章狀態與季節進行比較不會有任何編譯上的錯誤。

答案在 Go 內建庫或者一些咱們都知道的開源庫的程式碼裡就能找到。比如看看 google.golang.org/grpc/codes 裡的gRPC 的錯誤碼是怎麼定義的,就能明白該怎麼正確的實現列舉。

我們可以用 int 作為基礎型別建立一個別名型別,Go 裡邊是支援這個的

type Season int

const (
	Summer Season = 1
	Autumn        = 2
	Winter        = 3
	Spring        = 4
)

當然定義連續的常數值的時候 Go 裡邊經常使用 iota,所以上面的定義還能進一步簡化。

type Season int

const (
	Summer Season = iota + 1
	Autumn
	Winter
	Spring
)

type ArticleState int

const (
  Draft int = iota + 1
  Published
  Deleted  
)

func checkArticleState(state ArticleState) bool {
	// ... 
}

 func main() {
   // 兩個運算元型別不匹配,編譯錯誤
   fmt.Println(Autumn == Draft)
	// 引數型別不匹配,但是因為 ArticleState 底層的型別是 int 所以傳遞 int 的時候會發生隱式型別轉換,所以不會報錯
   checkArticleState(100)
 }

雖然這些狀態值的底層的型別都是 int 值,但是現在進行兩個不相干型別的列舉值比較,會造成編譯錯誤,因為現在我們使用狀態值的地方都有了型別限制。

不過函數 checkArticleState 的引數型別設定成了 ArticleState 但是因為 ArticleState 底層的型別是 int 。所以呼叫 checkArticleState 時傳遞 int 型別的引數會發生隱式型別轉換,不會造成編譯報錯,這塊如果想解決,只能重新定義型別來實現了,

可以參考StackOverflow上的這個答案:

https://stackoverflow.com/questions/50826100/how-to-disable-implicit-type-conversion-for-constants

更多程式設計相關知識,請存取:!!

以上就是淺析Golang中怎麼實現列舉(附程式碼)的詳細內容,更多請關注TW511.COM其它相關文章!