Go語言基礎之基本資料型別
Go語言基礎之基本資料型別
Go語言中有豐富的資料型別,除了基本的整型、浮點型、布林型、字串外,還有陣列、切片、結構體、函式、map、通道(channel)等。Go 語言的基本型別和其他語言大同小異。
基本資料型別
整型
整型分為以下兩個大類: 按長度分為:int8、int16、int32、int64 對應的無符號整型:uint8、uint16、uint32、uint64
其中,uint8
就是我們熟知的byte
型,int16
對應C語言中的short
型,int64
對應C語言中的long
型。
型別 | 描述 |
---|---|
uint8 | 無符號 8位整型 (0 到 255) 對應byte型別 |
uint16 | 無符號 16位整型 (0 到 65535) |
uint32 | 無符號 32位整型 (0 到 4294967295) |
uint64 | 無符號 64位整型 (0 到 18446744073709551615) |
int8 | 有符號 8位整型 (-128 到 127) |
int16 | 有符號 16位整型 (-32768 到 32767) |
int32 | 有符號 32位整型 (-2147483648 到 2147483647) 對應rune型別 |
int64 | 有符號 64位整型 (-9223372036854775808 到 9223372036854775807) |
特殊整型
型別 | 描述 |
---|---|
uint | 32位作業系統上就是uint32 ,64位作業系統上就是uint64 |
int | 32位作業系統上就是int32 ,64位作業系統上就是int64 |
uintptr | 無符號整型,用於存放一個指標 |
注意: 在使用int
和 uint
型別時,不能假定它是32位或64位的整型,而是考慮int
和uint
可能在不同平臺上的差異。預設值為0
注意事項 獲取物件的長度的內建len()
函式返回的長度可以根據不同平臺的位元組長度進行變化。實際使用中,切片或 map 的元素數量等都可以用int
來表示。在涉及到二進位制傳輸、讀寫檔案的結構描述時,為了保持檔案的結構不會受到不同編譯目標平臺位元組長度的影響,不要使用int
uint
。
數字字面量語法(Number literals syntax)
Go1.13版本之後引入了數字字面量語法,這樣便於開發者以二進位制、八進位制或十六進位制浮點數的格式定義數字,例如:
v := 0b00101101, 代表二進位制的 101101,相當於十進位制的 45。 v := 0o377,代表八進位制的 377,相當於十進位制的 255。 v := 0x1p-2,代表十六進位制的 1 除以 2²,也就是 0.25。 而且還允許我們用 _ 來分隔數字,比如說:
v := 123_456 等於 123456。
我們可以藉助fmt函式來將一個整數以不同進位制形式展示。
package main
import "fmt"
func main(){
// 十進位制
var a int = 10
fmt.Printf("%d \n", a) // 10
fmt.Printf("%b \n", a) // 1010 佔位符%b表示二進位制
// 八進位制 以0開頭
var b int = 077
fmt.Printf("%o \n", b) // 77
// 十六進位制 以0x開頭
var c int = 0xff
fmt.Printf("%x \n", c) // ff
fmt.Printf("%X \n", c) // FF
}
對於兩個不同型別的整數型不能直接比較,但各種型別的整型變數都可以直接與字面常量進行比較,如下
var i int32
var j int64
i, j = 1, 2
if i == j { //編譯錯誤
fmt.Println("i and j are equal.")
}
if i == 1 || j == 2 { //編譯通過
fmt.Println("i and j are equal.")
}
浮點型
Go語言支援兩種浮點型數:float32
和float64
。這兩種浮點型資料格式遵循IEEE 754
標準: float32
的浮點數的最大範圍約為 3.4e38
,可以使用常量定義:math.MaxFloat32
。 float64
的浮點數的最大範圍約為 1.8e308
,可以使用一個常量定義:math.MaxFloat64
。
列印浮點數時,可以使用fmt
包配合動詞%f
,程式碼如下:
package main
import (
"fmt"
"math"
)
func main() {
fmt.Printf("%f\n", math.Pi)
fmt.Printf("%.2f\n", math.Pi) // 保留2位小數(四捨五入),不夠在左側填充空格
}
浮點數比較 因為浮點數不是一種精確的表達方式,所以像整型那樣直接用==來判斷兩個浮點數是否相等 是不可行的,這可能會導致不穩定的結果,例如
package main
import (
"fmt"
)
func main() {
x := 0.1
y := 0.2
res := x + y
fmt.Println(res) // 0.30000000000000004
fmt.Println(res == 0.3) // false
}
浮點數裡有個叫非數的東西
fmt.Println(math.NaN()) // NaN
var z float32
fmt.Println(z) // 0
fmt.Println(z/z) // NaN
x:=z/z
fmt.Println(x < x) // false
fmt.Println(x > x) // false
fmt.Println(x == x) // false
fmt.Println(x != x) // true
複數
complex64和complex128
var c1 complex64
c1 = 1 + 2i
var c2 complex128
c2 = 2 + 3i
fmt.Println(c1)
fmt.Println(c2)
複數有實部和虛部,complex64的實部和虛部為32位,complex128的實部和虛部為64位。
布林值
Go語言中以bool
型別進行宣告布林型資料,布林型資料只有true(真)
和false(假)
兩個值。
注意:
- 布林型別變數的預設值為
false
。 - Go 語言中不允許將整型強制轉換為布林型.
- 布林型無法參與數值運算,也無法與其他型別進行轉換。
字串
Go語言中的字串以原生資料型別出現,使用字串就像使用其他原生資料型別(int、bool、float32、float64 等)一樣。 Go 語言裡的字串的內部實現使用UTF-8
編碼。 字串的值為雙引號(")
中的內容,可以在Go語言的原始碼中直接新增非ASCII碼字元,例如:
s1 := "hello"
s2 := "你好"
s1:='汪' // 字元型別
fmt.Println(s1)
fmt.Printf("%T",s1) // int32,即rune型別
字串轉義符
Go 語言的字串常見轉義符包含回車、換行、單雙引號、製表符等,如下表所示。
轉義符 | 含義 |
---|---|
\r |
回車符(返回行首) |
\n |
換行符(直接跳到下一行的同列位置) |
\t |
製表符 |
\' |
單引號 |
\" |
雙引號 |
\\ |
反斜槓 |
舉個例子,我們要列印一個Windows平臺下的一個檔案路徑:
package main
import (
"fmt"
)
func main() {
fmt.Println("str := \"c:\\Code\\lesson1\\go.exe\"")
}
多行字串
Go語言中要定義一個多行字串時,就必須使用反引號
字元:
s1 := `
第一行
第二行
第三行
`
fmt.Println(s1)
反引號間換行將被作為字串中的換行,但是所有的轉義字元均無效,文字將會原樣輸出。
字串的常用操作
方法 | 介紹 |
---|---|
len(str) | 求長度 |
+或fmt.Sprintf | 拼接字串 |
strings.Split | 分割 |
strings.contains | 判斷是否包含 |
strings.HasPrefix,strings.HasSuffix | 字首/字尾判斷 |
strings.Index(),strings.LastIndex() | 子串出現的位置 |
strings.Join(a[]string, sep string) | join操作 |
len(str) 求長度(位元組個數,文字字符采用的是utf-8編碼)
例如:len("a你") 結果為4
// 字串拼接
+或fmt.Sprintf拼接字串
例如:d := fmt.Sprintf("%v - %v - %v", "2020", "05", 20) 結果為:2020 - 05 - 20
// strings
strings.HasPrefix(s string,preffix string) bool:
判斷字串s是否以prefix開頭
stirngs.HasSuffix(s string,suffix string) bool:
判斷字串s是否以suffix結尾
strings.Index(s string,str string) int:
判斷str在s中首次出現的位置,如果沒有出現,則返回-1
strings.LastIndex(s string,str string) int:
判斷str在s中最後出現的位置,如果沒有出現,則返回-1
strings.Replace(str string,old string,new string,n int):
字串替換
strings.Count(str string,count int)string:
字串計數
strings.Repeat(str string,count int) string:
重複count次str
strings.ToLower(str string)
轉換為小寫
strings.ToUpper(str string)string:
轉換為大寫
strings.TrimSpace(str string):
去掉字串首位空白字元
strings.Trim(str string,cut string):
去掉字串首尾cut字元
strings.TrimLeft(str string,cut string):
去掉字串首部cut字元
strings.TrimRight(str string,cunt string):
去掉字串尾部cut字元
strings.Field(str string):
返回str空格分隔的所有子串的slice
string.Split(str string,split string):
返回str split分割的所有子串的slice
例如:strings.Split("randy:18:male",":")結果為切片:[randy 18 male]
strings.Join(s1 []string,sep string):
用sep把s1中的所有元素連線起來
例如:strings.Join([]string{"randy","18","male"},":")結果為randy:18:male
strings.contains
判斷是否包含
例如:strings.Contains("abcdef","abc")結果為true
// strconv
scronv.Itoa(i int):把一個整數轉換成字串
scronv.Atio(str string)(int,errror):
把一個字串轉換成整數
byte和rune型別
組成每個字串的元素叫做字元
,可以通過遍歷或者單個獲取字串元素獲得字元。 字元用單引號(’)包裹起來,如:
var a := '中'
var b := 'x'
Go 語言的字元有以下兩種:
uint8
型別,或者叫 byte 型,代表了ASCII碼
的一個字元。rune
型別,代表一個UTF-8字元
。
當需要處理中文、日文或者其他複合字符時,則需要用到rune
型別。rune
型別實際是一個int32
。
Go 使用了特殊的 rune 型別來處理 Unicode,讓基於 Unicode 的文字處理更為方便,也可以使用 byte 型進行預設字串處理,效能和擴充套件性都有照顧。
// 遍歷字串
func traversalString() {
s := "hello沙河"
for i := 0; i < len(s); i++ { //byte
fmt.Printf("%v(%c) ", s[i], s[i])
}
fmt.Println()
for _, r := range s { //rune
fmt.Printf("%v(%c) ", r, r)
fmt.Printf("%c", r) // 字元使用`%c`引數列印,或者是用`%q`引數列印帶單引號的字元:
}
fmt.Println()
}
輸出:
104(h) 101(e) 108(l) 108(l) 111(o) 230(æ) 178(²) 153() 230(æ) 178(²) 179(³)
104(h) 101(e) 108(l) 108(l) 111(o) 27801(沙) 27827(河)
因為UTF8編碼下一個中文漢字由3~4個位元組組成,所以我們不能簡單的按照位元組去遍歷一個包含中文的字串,否則就會出現上面輸出中第一行的結果。
字串底層是一個byte陣列,所以可以和[]byte
型別相互轉換。字串是不能修改的 字串是由byte位元組組成,所以字串的長度是byte位元組的長度。 rune型別用來表示utf8字元,一個rune字元由一個或多個byte組成。
動 詞 | 功 能 |
---|---|
%v | 按值的本來值輸出 |
%+v | 在 %v 基礎上,對結構體欄位名和值進行展開 |
%#v | 輸出 Go 語言語法格式的值 |
%T | 輸出 Go 語言語法格式的型別和值 |
%% | 輸出 % 本體 |
%b | 整型以二進位制方式顯示 |
%o | 整型以八進位制方式顯示 |
%d | 整型以十進位制方式顯示 |
%x | 整型以十六進位制方式顯示 |
%X | 整型以十六進位制、字母大寫方式顯示 |
%U | Unicode 字元 |
%f | 浮點數 |
%p | 指標,十六進位制方式顯示 |
修改字串
要修改字串,需要先將其轉換成[]rune
或[]byte
,完成後再轉換為string
。無論哪種轉換,都會重新分配記憶體,並複製位元組陣列。
func changeString() {
s1 := "big"
// 強制型別轉換
byteS1 := []byte(s1)
byteS1[0] = 'p'
fmt.Println(string(byteS1))
s2 := "白蘿蔔"
runeS2 := []rune(s2)
runeS2[0] = '紅'
fmt.Println(string(runeS2))
}
pig
紅蘿蔔
型別轉換
Go語言中只有強制型別轉換,沒有隱式型別轉換。該語法只能在兩個型別之間支援相互轉換的時候使用。
強制型別轉換的基本語法如下:
T(表示式)
其中,T表示要轉換的型別。表示式包括變數、複雜運算元和函式返回值等.
比如計算直角三角形的斜邊長時使用math包的Sqrt()函式,該函式接收的是float64型別的引數,而變數a和b都是int型別的,這個時候就需要將a和b強制型別轉換為float64型別。
func sqrtDemo() {
var a, b = 3, 4
var c int
// math.Sqrt()接收的引數是float64型別,需要強制轉換
c = int(math.Sqrt(float64(a*a + b*b)))
fmt.Println(c)
}
總結
- 整型 int 預設值為0
- 浮點型 float
- 字元型別:位元組 byte:-uint8的別名,rune-int32的別名,通過型別轉換修改字串
- 複數complex64
- 字串 string 預設值為空字串
- 布林型別bool預設值為false
- 型別轉換T(表示式)