溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點(diǎn)擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

go語言中的type是什么

發(fā)布時(shí)間:2020-06-18 10:40:11 來源:億速云 閱讀:393 作者:Leah 欄目:編程語言

go語言中的type是什么?可能很多人都不太了解,為了讓大家更加了解,小編給大家總結(jié)了以下內(nèi)容,希望大家根據(jù)這篇文章可以有所收獲。

type是go語法里的重要而且常用的關(guān)鍵字,type絕不只是對應(yīng)于C/C++中的typedef。type又有兩種使用方式,一種是類型別名,一種是類型定義,是不是熟悉的C語言(define和typedef)味道?

類型定義

type Student struct {
  name String
  age int
}
type I int

類型別名

type Sdt = Student
type I = int

type有如下幾種用法:

定義結(jié)構(gòu)體

定義接口

類型定義

類型別名

類型查詢

定義結(jié)構(gòu)體

結(jié)構(gòu)體是用戶自定義的一種抽象的數(shù)據(jù)結(jié)構(gòu),golang中struct類似于java語言中的class, 在程序設(shè)計(jì)中,有著舉足輕重的地位。結(jié)構(gòu)體的用法,將會(huì)在struct關(guān)鍵字中詳細(xì)的介紹。下邊來看一下定義一個(gè)結(jié)構(gòu)體的語法格式:

type name struct {
    Field1  dataType
    Field2  dataType
    Field3  dataType
}

定義接口

接口相關(guān)知識(shí)點(diǎn),將會(huì)在interface關(guān)鍵字中詳細(xì)介紹,下邊來看一段定義接口的語法格式:

type name interface{
    Read()
    Write()
}

類型定義

使用類型定義定義出來的類型與原類型不相同,所以不能使用新類型變量賦值給原類型變量,除非使用強(qiáng)制類型轉(zhuǎn)換。下面來看一段示例代碼,根據(jù)string類型,定義一種新的類型,新類型名稱是name:

type name string

為什么要使用類型定義呢?

類型定義可以在原類型的基礎(chǔ)上創(chuàng)造出新的類型,有些場合下可以使代碼更加簡潔,如下邊示例代碼:

package main
import (
  "fmt"
)
// 定義一個(gè)接收一個(gè)字符串類型參數(shù)的函數(shù)類型
type handle func(str string)
// exec函數(shù),接收handle類型的參數(shù)
func exec(f handle) {
  f("hello")
}
func main() {
  // 定義一個(gè)函數(shù)類型變量,這個(gè)函數(shù)接收一個(gè)字符串類型的參數(shù)
  var p = func(str string) {
    fmt.Println("first", str)
  }
  exec(p)
  // 匿名函數(shù)作為參數(shù)直接傳遞給exec函數(shù)
  exec(func(str string) {
    fmt.Println("second", str)
  })
}

輸出信息是:

first hello
second hello

上邊的示例是類型定義的一種簡單應(yīng)用場合,如果不使用類型定義,那么想要實(shí)現(xiàn)上邊示例中的功能,應(yīng)該怎么書寫這段代碼呢?

// exec函數(shù),接收handle類型的參數(shù)
func exec(f func(str string)) {
  f("hello")
}

exec函數(shù)中的參數(shù)類型,需要替換成func(str string)了,咋一看去也不復(fù)雜,但是假如exec接收一個(gè)需要5個(gè)參數(shù)的函數(shù)變量呢?是不是感覺參數(shù)列表就會(huì)很長了。

func exec(f func(str string, str2 string, num int, money float64, flag bool)) {
  f("hello")
}

從上邊的代碼可以發(fā)現(xiàn),exec函數(shù)的參數(shù)列表可讀性變差了。下邊再來看看使用類型定義是怎么實(shí)現(xiàn)這個(gè)功能:

package main
import (
  "fmt"
)
// 定義一個(gè)需要五個(gè)參數(shù)的函數(shù)類型
type handle func(str string, str2 string, num int, money float64, flag bool)
// exec函數(shù),接收handle類型的參數(shù)
func exec(f handle) {
  f("hello", "world", 10, 11.23, true)
}
func demo(str string, str2 string, num int, money float64, flag bool) {
  fmt.Println(str, str2, num, money, flag)
}
func main() {
  exec(demo)
}

類型別名

類型別名這個(gè)特性在golang1.9中引入。使用類型別名定義出來的類型與原類型一樣,即可以與原類型變量互相賦值,又擁有了原類型的所有方法集。給strng類型取一個(gè)別名,別名名稱是name:

type name = string

類型別名與類型定義不同之處在于,使用類型別名需要在別名和原類型之間加上賦值符號(=);使用類型別名定義的類型與原類型等價(jià),而使用類型定義出來的類型是一種新的類型。

如下邊示例:

package main
import (
  "fmt"
)
type a = string
type b string
func SayA(str a) {
  fmt.Println(str)
}
func SayB(str b) {
  fmt.Println(str)
}
func main() {
  var str = "test"
  SayA(str)
  
  //錯(cuò)誤參數(shù)傳遞,str是字符串類型,不能賦值給b類型變量
  SayB(str)
}

這段代碼在編譯時(shí)會(huì)出現(xiàn)如下錯(cuò)誤:

.\main.go:21:6: cannot use str (type string) as type b in argument to SayB

從錯(cuò)誤信息可知,str為字符串類型,不能當(dāng)做b類型參數(shù)傳入SayB函數(shù)中。而str卻可以當(dāng)做a類型參數(shù)傳入到SayA函數(shù)中。由此可見,使用類型別名定義的類型與原類型一致,而類型定義定義出來的類型,是一種新的類型。

給類型別名新增方法,會(huì)添加到原類型方法集中

給類型別名新增方法后,原類型也能使用這個(gè)方法。下邊請看一段示例代碼:

package main
import (
  "fmt"
)
// 根據(jù)string類型,定義類型S
type S string
func (r *S) Hi() {
  fmt.Println("S hi")
}
// 定義S的類型別名為T
type T = S
func (r *T) Hello() {
  fmt.Println("T hello")
}
// 函數(shù)參數(shù)接收S類型的指針變量
func exec(obj *S) {
  obj.Hello()
  obj.Hi()
}
func main() {
  t := new(T)
  s := new(S)
  exec(s)
  // 將T類型指針變量傳遞給S類型指針變量
  exec(t)
}

輸出信息是:

T hello
S hi
T hello
S hi

上邊的示例中,S是原類型,T是S類型別名。在給T增加了Hello方法后,S類型的變量也可以使用Hello方法。說明給類型別名新增方法后,原類型也能使用這個(gè)方法。從示例中可知,變量t可以賦值給S類型變量s,所以類型別名是給原類型取了一個(gè)小名,本質(zhì)上沒有發(fā)生任何變化。

類型別名,只能對同一個(gè)包中的自定義類型產(chǎn)生作用。舉個(gè)例子,golang sdk中有很多個(gè)包,是不是我們可以使用類型別名,給sdk包中的結(jié)構(gòu)體類型新增方法呢?答案是:不行。請牢記一點(diǎn):類型別名,只能對包內(nèi)的類型產(chǎn)生作用,對包外的類型采用類型別名,在編譯時(shí)將會(huì)提示如下信息:

cannot define new methods on non-local type string

類型查詢

類型查詢,就是根據(jù)變量,查詢這個(gè)變量的類型。為什么會(huì)有這樣的需求呢?goalng中有一個(gè)特殊的類型interface{},這個(gè)類型可以被任何類型的變量賦值,如果想要知道到底是哪個(gè)類型的變量賦值給了interface{}類型變量,就需要使用類型查詢來解決這個(gè)需求,示例代碼如下:

package main
import (
  "fmt"
)
func main() {
    // 定義一個(gè)interface{}類型變量,并使用string類型值”abc“初始化
    var a interface{} = "abc"
    
    // 在switch中使用 變量名.(type) 查詢變量是由哪個(gè)類型數(shù)據(jù)賦值。
    switch v := a.(type) {
    case string:
      fmt.Println("字符串")
    case int:
        fmt.Println("整型")
    default:
      fmt.Println("其他類型", v)
    }
}

如果使用.(type)查詢類型的變量不是interface{}類型,則在編譯時(shí)會(huì)報(bào)如下錯(cuò)誤:

cannot type switch on non-interface value a (type string)

如果在switch以外地方使用.(type),則在編譯時(shí)會(huì)提示如下錯(cuò)誤:

use of .(type) outside type switch

所以,使用type進(jìn)行類型查詢時(shí),只能在switch中使用,且使用類型查詢的變量類型必須是interface{}

看完上述內(nèi)容,你們對go語言中的type有進(jìn)一步的了解嗎?如果還想了解更多相關(guān)內(nèi)容,歡迎關(guān)注億速云行業(yè)資訊頻道,感謝各位的閱讀。

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI