GO語言系列(四)- 內置函數、閉包與高級數據類型

1、內置函數、遞歸函數、閉包

內置函數

  • 1. close:主要用來關閉channel
  • 2. len:用來求長度,好比string、array、slice、map、channel
  • 3. new:用來分配內存,主要用來分配值類型,好比int、struct。返回的是指針
  • 4. make:用來分配內存,主要用來分配引用類型,好比chan、map、slice
  • 5. append:用來追加元素到數組、slice中
  • 6. panic和recover:用來作錯誤處理
  • 7. new和make的區別

package main

import (
    "errors"
    "fmt"
)

func initConfig() (err error) {
    return errors.New("init config failed")
}

func test() {
    /*
        defer func() {
            if err := recover(); err != nil {
                fmt.Println(err)
            }
        }()
    */
    err := initConfig()
    if err != nil {
        panic(err)
    }
    return
}

func main() {
    test()
    var i int
    fmt.Println(i)  // 0
    fmt.Println(&i) // 0xc0000100a8

    j := new(int)
    *j = 100
    fmt.Println(j)  // 0xc00005a088
    fmt.Println(*j) // 100

    var a []int // 定義slice
    a = append(a, 10, 20, 354)
    a = append(a, a...) // ...可變參數   slice展開
    fmt.Println(a)
}
內置函數案例
package main

import "fmt"

func test() {
    s1 := new([]int)
    fmt.Println(s1)

    s2 := make([]int, 10)
    fmt.Println(s2)

    *s1 = make([]int, 5)
    (*s1)[0] = 100
    s2[0] = 100
    fmt.Println(s1)
    fmt.Println(s2)
    return
}

func main() {
    test()
}
內置函數2
func main() {
    testError()
    afterErrorfunc()
}

func testError() {
    defer func() {
        if r := recover(); r != nil {
            fmt.Println("testError() 遇到錯誤:", r)
        }
    }()

    panic(" \"panic 錯誤\"")
    fmt.Println("拋出一個錯誤後繼續執行代碼")
}

func afterErrorfunc() {
    fmt.Println("遇到錯誤以後 func ")
}
panic和recover
func main() {
    err := testError()
    if err != nil {
        fmt.Println("main 函數獲得錯誤類型:", err)
    }
    afterErrorfunc()
}

func testError() (err error) {
    defer func() {
        if r := recover(); r != nil {
            fmt.Println("testError() 遇到錯誤:", r)
            switch x := r.(type) {
            case string:
                err = errors.New(x)
            case error:
                err = x
            default:
                err = errors.New("")
            }
        }
    }()
    panic(" \"panic 錯誤\"")
    fmt.Println("拋出一個錯誤後繼續執行代碼")
    return nil
}

func afterErrorfunc() {
    fmt.Println("遇到錯誤以後 func ")
}
panic和recover有返回值

遞歸函數

  • 1. 一個函數調用本身,就叫作遞歸。python

  • 2. 遞歸設計原則mysql

1)一個大的問題可以分解成類似的小問題git

2)定義好出口條件github

 案例:遞歸輸出hello,實現階乘,實現斐波那契數

package main

import (
	"fmt"
	"time"
)

// 遞歸
func recusive(n int) {
	fmt.Println("hello", n)
	time.Sleep(time.Second)
	if n > 10 {
		return
	}
	recusive(n + 1) // 至少上百萬上千萬次遞歸
}

// 階乘
func factor(n int) int {
	if n == 1 {
		return 1
	}
	return factor(n-1) * n
}

//斐波那契數
func fab(n int) int {
	if n <= 1 {
		return 1
	}
	return fab(n-1) + fab(n-2)
}

func main() {
	// fmt.Println(factor(5))
	recusive(0)
	// for i := 0; i < 10; i++ {
	// fmt.Println(fab(i))
	// }
}

 閉包

閉包:一個函數和與其相關的引用環境組合而成的實體golang

案例:Adder和爲文件名添加後綴sql

package main

import (
	"fmt"
	"strings"
)

func Adder() func(int) int {
	var x int
	return func(d int) int {
		x += d
		return x
	}
}

func makeSuffix(suffix string) func(string) string {
	return func(name string) string {
		if strings.HasSuffix(name, suffix) == false {
			return name + suffix
		}
		return name
	}
}

func main() {
	f := Adder()
	fmt.Println(f(1))
	fmt.Println(f(100))
	fmt.Println(f(1000))

	f1 := makeSuffix(".bmp")
	fmt.Println(f1("test"))
	fmt.Println(f1("pic"))

	f2 := makeSuffix(".jpg")
	fmt.Println(f2("test"))
	fmt.Println(f2("pic"))
}

2、數組和切片

數組

  • 1. 數組:是同一種數據類型的固定長度的序列。
  • 2. 數組定義:var a [len]int,好比:var a[5]int,一旦定義,長度不能變
  • 3. 長度是數組類型的一部分,所以,var a[5] int和var a[10]int是不一樣的類型
  • 4. 數組能夠經過下標進行訪問,下標是從0開始,最後一個元素下標是:len-1
for i := 0; i < len(a); i++ {
}
for index, v := range a {
}
  • 5. 訪問越界,若是下標在數組合法範圍以外,則觸發訪問越界,會panic
  • 6. 數組是值類型,所以改變副本的值,不會改變自己的值
package main

import "fmt"

func test1() {
    var a [10]int
    a[0] = 10
    a[9] = 100
    fmt.Println(a)

    for i := 0; i < len(a); i++ {
        fmt.Println(a[i])
    }
    for index, val := range a {
        fmt.Printf("a[%d]=%d\n", index, val)
    }
}

func test2() {
    var a [10]int
    b := a
    b[0] = 100
    fmt.Println(a)
}

func test3(arr *[5]int) {
    (*arr)[0] = 1000
}

func main() {
    test1()
    test2()
    var a [5]int
    test3(&a)
    fmt.Println(a)
}

/*
[10 0 0 0 0 0 0 0 0 100]
10
0
0
0
0
0
0
0
0
100
a[0]=10
a[1]=0
a[2]=0
a[3]=0
a[4]=0
a[5]=0
a[6]=0
a[7]=0
a[8]=0
a[9]=100
[0 0 0 0 0 0 0 0 0 0]
[1000 0 0 0 0]
*/
數組示例一

數組初始化

a. var age0 [5]int = [5]int{1,2,3}
b. var age1 = [5]int{1,2,3,4,5}
c. var age2 = […]int{1,2,3,4,5,6}
d. var str = [5]string{3:」hello world」, 4:」tom」}

多維數組

a. var age [5][3]int
b. var f [2][3]int = [...][3]int{{1, 2, 3}, {7, 8, 9}}  

多維數組遍歷

package main

import (
	"fmt"
)

func main() {

	var f [2][3]int = [...][3]int{{1, 2, 3}, {7, 8, 9}}

	for k1, v1 := range f {
		for k2, v2 := range v1 {
			fmt.Printf("(%d,%d)=%d ", k1, k2, v2)
		}
		fmt.Println()
	}
} 

數組示例數組

package main

import (
    "fmt"
)

func fab(n int) {
    var a []int
    a = make([]int, n)

    a[0] = 1
    a[1] = 1

    for i := 2; i < n; i++ {
        a[i] = a[i-1] + a[i-2]
    }

    for _, v := range a {
        fmt.Println(v)
    }
}

func testArray() {
    var a [5]int = [5]int{1, 2, 3, 4, 5}
    var a1 = [5]int{1, 2, 3, 4, 5}
    var a2 = [...]int{38, 283, 48, 38, 348, 387, 484}
    var a3 = [...]int{1: 100, 3: 200}
    var a4 = [...]string{1: "hello", 3: "world"}

    fmt.Println(a)
    fmt.Println(a1)
    fmt.Println(a2)
    fmt.Println(a3)
    fmt.Println(a4)
    /*
        [1 2 3 4 5]
        [1 2 3 4 5]
        [38 283 48 38 348 387 484]
        [0 100 0 200]
        [ hello  world]
    */
}

func testArray2() {
    var a [2][5]int = [...][5]int{{1, 2, 3, 4, 5}, {6, 7, 8, 9, 10}}

    for row, v := range a {
        for col, v1 := range v {
            fmt.Printf("(%d,%d)=%d ", row, col, v1)
        }
        fmt.Println()
    }
}

func main() {
    fab(10)
    testArray()
    testArray2()
}
數組示例二

切片

1. 切片相關概念

  • 1. 切片:切片是數組的一個引用,所以切片是引用類型
  • 2. 切片的長度能夠改變,所以,切片是一個可變的數組
  • 3. 切片遍歷方式和數組同樣,能夠用len()求長度
  • 4. cap能夠求出slice最大的容量,0 <= len(slice) <= (array),其中array是slice引用的數組
  • 5. 切片的定義:var 變量名 []類型,好比 var str []string var arr []int

2. 切片的相關語法

  • 1. 切片初始化:var slice []int = arr[start:end] 包含start到end之間的元素,但不包含end
  • 2. Var slice []int = arr[0:end]能夠簡寫爲 var slice []int=arr[:end]
  • 3. Var slice []int = arr[start:len(arr)] 能夠簡寫爲 var slice[]int = arr[start:]
  • 4. Var slice []int = arr[0, len(arr)] 能夠簡寫爲 var slice[]int = arr[:]
  • 5. 若是要切片最後一個元素去掉,能夠這麼寫: Slice = slice[:len(slice)-1]

3. 切片的內存佈局

4. 經過make來建立切片

var slice []type = make([]type, len)
slice := make([]type, len)
slice := make([]type, len, cap)

5. 用append內置函數操做切片

slice = append(slice, 10)
var a = []int{1,2,3}
var b = []int{4,5,6}
a = append(a, b…)

6. For range 遍歷切片

for index, val := range slice {
}

7. 切片resize

var a = []int {1,3,4,5}
b := a[1:2]
b = b[0:3]

8. 切片拷貝

s1 := []int{1,2,3,4,5}

s2 := make([]int, 10)

copy(s2, s1)

s3 := []int{1,2,3}

s3 = append(s3, s2…)

s3 = append(s3, 4,5,6)

9. string與slice

  string底層就是一個byte的數組,所以,也能夠進行切片操做數據結構

str := 「hello world」
s1 := str[0:5]
fmt.Println(s1)
s2 := str[5:]
fmt.Println(s2)

10. string的底層佈局

11. 如何改變string中的字符值?

string自己是不可變的,所以要改變string中字符,須要以下操做:閉包

str := 「hello world」
s := []byte(str)
s[0] = ‘o’
str = string(s)

12. 排序和查找操做

排序操做主要都在 sort包中,導入就可使用了app

sort.Ints對整數進行排序, sort.Strings對字符串進行排序, sort.Float64s對
浮點數進行排序.
sort.SearchInts(a []int, b int) 從數組a中查找b,前提是a必須有序
sort.SearchFloats(a []float64, b float64) 從數組a中查找b,前提是a必須有序
sort.SearchStrings(a []string, b string) 從數組a中查找b,前提是a必須有序

示例

package main

import "fmt"

type slice struct {
    ptr *[100]int
    len int
    cap int
}

func make1(s slice, cap int) slice {
    s.ptr = new([100]int)
    s.cap = cap
    s.len = 0
    return s
}

func testSlice() {
    var slice []int
    var arr [5]int = [...]int{1, 2, 3, 4, 5}

    slice = arr[:]
    fmt.Println(slice)
    fmt.Println(arr[2:4]) // [3,4]
    fmt.Println(arr[2:])  // [3,4,5]
    fmt.Println(arr[0:1]) // [1]
    fmt.Println(arr[:len(arr)-1])
}

func modify(s slice) {
    s.ptr[1] = 1000
}

func testSlice2() {
    var s1 slice
    s1 = make1(s1, 10)

    s1.ptr[0] = 100
    modify(s1)

    fmt.Println(s1.ptr)
}

func modify1(a []int) {
    a[1] = 1000
}

func testSlice3() {
    var b []int = []int{1, 2, 3, 4}
    modify1(b)
    fmt.Println(b)
}

func testSlcie4() {
    var a = [10]int{1, 2, 3, 4}

    b := a[1:5]
    fmt.Printf("%p\n", b)     // 0xc000014238
    fmt.Printf("%p\n", &a[1]) // 0xc000014238
}

func main() {
    // testSlice()
    testSlice2()
    testSlice3()
    testSlcie4()
}
切片的用法示例一
package main

import "fmt"

func testSlice() {
    var a [5]int = [...]int{1, 2, 3, 4, 5}
    s := a[1:]
    fmt.Println("a:", a)
    s[1] = 100
    fmt.Printf("s=%p a[1]=%p\n", s, &a[1])
    fmt.Println("before a:", a)

    s = append(s, 10)
    s = append(s, 10)
    s = append(s, 10)
    s = append(s, 10)
    s = append(s, 10)

    s[1] = 1000
    fmt.Println("after a:", a)
    fmt.Println(s)
    fmt.Printf("s=%p a[1]=%p\n", s, &a[1])
}

func testCopy() {
    var a []int = []int{1, 2, 3, 4, 5}
    b := make([]int, 10)

    copy(b, a)

    fmt.Println(b)
}

func testString() {
    s := "hello world"
    s1 := s[0:5]
    s2 := s[6:]

    fmt.Println(s1)
    fmt.Println(s2)
}

func testModifyString() {
    s := "我hello world"
    s1 := []rune(s)

    s1[0] = 200
    s1[1] = 128
    s1[2] = 256
    str := string(s1)
    fmt.Println(str)
}

func main() {
    // testSlice()
    testCopy()
    testString()
    testModifyString()
}
切片的用法示例二
package main

import (
    "fmt"
    "sort"
)

func testIntSort() {
    var a = [...]int{1, 8, 43, 2, 456}
    sort.Ints(a[:])
    fmt.Println(a)
}

func testStrings() {
    var a = [...]string{"abc", "efg", "b", "A", "eeee"}
    sort.Strings(a[:])

    fmt.Println(a)
}

func testFloat() {
    var a = [...]float64{2.3, 0.8, 28.2, 392342.2, 0.6}
    sort.Float64s(a[:])
    fmt.Println(a)
}

func testIntSearch() {
    var a = [...]int{1, 8, 43, 2, 456}
    index := sort.SearchInts(a[:], 2)
    fmt.Println(index)
}

func main() {
    testIntSort()
    testStrings()
    testFloat()
    testIntSearch()
}
切片的用法示例三

3、map數據結構

1.map簡介

key-value的數據結構,又叫字典或關聯數組
a.聲明

var map1 map[keytype]valuetype
var a map[string]string
var a map[string]int
var a map[int]string
var a map[string]map[string]string

聲明是不會分配內存的,初始化須要make

2. map相關操做

var a map[string]string = map[string]string{「hello」: 「world」}
a = make(map[string]string, 10)
a[「hello」] = 「world」	// 插入和更新
Val, ok := a[「hello」]	//查找
for k, v := range a {	//遍歷
fmt.Println(k,v)
}
delete(a, 「hello」)	// 刪除
len(a)	// 長度

3. map是引用類型

func modify(a map[string]int) {
a[「one」] = 134
}

4. slice of map

Items := make([]map[int][int], 5)
For I := 0; I < 5; i++ {
items[i] = make(map[int][int])
}

5. map排序

a. 先獲取全部key,把key進行排序
b. 按照排序好的key,進行遍歷

6. Map反轉

  • a. 初始化另一個map,把key、value互換便可

 示例

package main

import "fmt"

func trans(a map[string]map[string]string) {
    for k, v := range a {
        fmt.Println(k)
        for k1, v1 := range v {
            fmt.Println("\t", k1, v1)
        }
    }
}

func testMap() {
    var a map[string]string = map[string]string{
        "key": "value",
    }
    // a := make(map[string]string, 10)
    a["abc"] = "efg"
    a["abc1"] = "wew"
    fmt.Println(a)
}

func testMap2() {
    a := make(map[string]map[string]string, 100)
    a["key1"] = make(map[string]string)
    a["key1"]["key2"] = "val2"
    a["key1"]["key3"] = "val3"
    a["key1"]["key4"] = "val4"
    a["key1"]["key5"] = "val5"
    a["key1"]["key6"] = "val6"
    fmt.Println(a)

}

func modify(a map[string]map[string]string) {
    _, ok := a["zhangsan"]
    if !ok {
        a["zhangsan"] = make(map[string]string)
    }
    a["zhangsan"]["pwd"] = "123456"
    a["zhangsan"]["nickname"] = "superman"
    return
}

func testMap3() {
    a := make(map[string]map[string]string, 100)
    modify(a)
    fmt.Println(a)
}

func testMap4() {
    a := make(map[string]map[string]string, 100)
    a["key1"] = make(map[string]string)
    a["key1"]["key2"] = "val2"
    a["key1"]["key3"] = "val3"
    a["key1"]["key4"] = "val4"
    a["key1"]["key5"] = "val5"
    a["key2"] = make(map[string]string)
    a["key2"]["key22"] = "val22"
    a["key2"]["key23"] = "val23"
    trans(a)
    delete(a, "key1")
    fmt.Println()
    trans(a)
}

func testMap5() {
    var a []map[int]int
    a = make([]map[int]int, 5)

    if a[0] == nil {
        a[0] = make(map[int]int)
    }
    a[0][10] = 10
    fmt.Println(a)
}

func main() {
    testMap()
    testMap2()
    testMap3()
    testMap4()
    testMap5()
}
map示例
package main

import (
    "fmt"
    "sort"
)

func testMapSort() {
    var a map[int]int
    a = make(map[int]int, 5)

    a[8] = 10
    a[3] = 10
    a[2] = 10
    a[1] = 10
    a[18] = 10

    var keys []int
    for k, _ := range a {
        keys = append(keys, k)
        // fmt.Println(k, v)
    }
    sort.Ints(keys)
    for _, v := range keys {
        fmt.Println(v, a[v])
    }
}

func testMapSort2() {
    var a map[string]int
    var b map[int]string

    a = make(map[string]int, 5)
    b = make(map[int]string, 5)
    a["8"] = 10
    a["3"] = 11
    a["2"] = 12
    a["1"] = 13
    a["18"] = 14

    for k, v := range a {
        b[v] = k
    }
    fmt.Println(b)
}

func main() {
    testMapSort()
    testMapSort2()
}
map示例2

4、包

1. golang中的包

  • a. golang目前有150個標準的包,覆蓋了幾乎全部的基礎庫

  • b. golang.org有全部包的文檔,沒事都翻翻

2. 線程同步

a. import(「sync」)
b. 互斥鎖, var mu sync.Mutex
c. 讀寫鎖, var mu sync.RWMutex
package main

import (
    "fmt"
    "math/rand"
    "sync"
    "sync/atomic"
    "time"
)

var rwLock *sync.RWMutex

func testLock() {

    var a map[int]int
    a = make(map[int]int, 5)

    a[8] = 10
    a[3] = 10
    a[2] = 10
    a[1] = 10
    a[18] = 10

    for i := 0; i < 2; i++ {
        go func(b map[int]int) {
            rwLock.RLock()
            b[8] = rand.Intn(100)
            rwLock.Unlock()
        }(a)
    }

    rwLock.RLock()
    fmt.Println(a)
    rwLock.Unlock()
}

func testRWLock() {
    // var rwLock myLocker = new(sync.RWMutex)
    // var rwLock sync.RWMutex
    // var rwLock sync.Mutex
    var a map[int]int
    a = make(map[int]int, 5)

    var count int32

    a[8] = 10
    a[3] = 10
    a[2] = 10
    a[1] = 10
    a[18] = 10

    for i := 0; i < 2; i++ {
        go func(b map[int]int) {
            rwLock.Lock()
            // lock.Lock()
            b[8] = rand.Intn(100)
            time.Sleep(time.Millisecond)
            rwLock.Unlock()
            // lock.Unlock()
        }(a)
    }

    for i := 0; i < 100; i++ {
        go func(b map[int]int) {
            for {
                rwLock.Lock()
                time.Sleep(time.Millisecond)
                // fmt.Println(a)
                rwLock.Unlock()
                atomic.AddInt32(&count, 1)
            }
        }(a)
    }

    time.Sleep(time.Second * 3)
    fmt.Println(atomic.LoadInt32(&count))
}

func main() {
    rwLock = new(sync.RWMutex)
    // testLock()
    testRWLock()
}
線程同步鎖示例

3. go get安裝第三方包

go get github.com/go-sql-driver/mysql 

 

本節做業

  • 1. 冒泡排序

  • 2. 選擇排序

  • 3. 插入排序

  • 4.快速排序

參考

package main

import "fmt"

// 冒泡排序:本質上是交換排序的一種
func bsort(a []int) {
    for i := 0; i < len(a); i++ {
        for j := 1; j < len(a)-i; j++ {
            if a[j] < a[j-1] {
                a[j], a[j-1] = a[j-1], a[j]
            }
        }
    }
}

func main() {
    b := [...]int{8, 7, 4, 5, 3, 2, 1}
    bsort(b[:])
    fmt.Println(b)
}
冒泡排序
package main

import "fmt"

// 選擇排序
func ssort(a []int) {
    for i := 0; i < len(a); i++ {
        var min int = i
        for j := i + 1; j < len(a); j++ {
            if a[min] > a[j] {
                min = j
            }
        }
        if min != i {
            a[i], a[min] = a[min], a[i]
        }
    }
}

func main() {
    b := [...]int{8, 7, 4, 5, 3, 2, 1}
    ssort(b[:])
    fmt.Println(b)
}
選擇排序
package main

import "fmt"

// 插入排序, 每次將一個數插入到有序序列當中合適的位置
func isort(a []int) {
    for i := 1; i < len(a); i++ {
        for j := i; j > 0; j-- {
            if a[j] > a[j-1] {
                break
            }
            a[j], a[j-1] = a[j-1], a[j]
        }
    }
}

func main() {
    b := [...]int{8, 7, 4, 5, 3, 2, 1}
    isort(b[:])
    fmt.Println(b)
}
插入排序
package main

import "fmt"

// 快速排序, 一次排序肯定一個元素的位置, 使左邊的元素都比它小,右邊的元素都比它大
func qsort(a []int, left, right int) {
    if left >= right {
        return
    }
    val := a[left]
    // 肯定val所在的位置
    k := left
    for i := left + 1; i <= right; i++ {
        if a[i] < val {
            a[k] = a[i]
            a[i] = a[k+1]
            k++
        }
    }
    a[k] = val
    qsort(a, left, k-1)
    qsort(a, k+1, right)
}

func main() {
    b := [...]int{8, 7, 4, 5, 3, 2, 1}
    qsort(b[:], 0, len(b)-1)
    fmt.Println(b)
}
快速排序
相關文章
相關標籤/搜索