Go 语言编程实例(四)

互斥体

在前面的例子中,我们看到了如何使用原子操作来管理简单的计数器状态。对于更复杂的状态,可以使用 互斥体 来安全地访问多个 goroutine 中的数据。

在这个例子中,状态(state)是一个映射。示例中的互斥将同步访问状态。我们将跟踪执行的读写操作数量。

这里启动 100 个 goroutine 来对状态执行重复读取,每个 goroutine 中每 ms 读取一次。对于每个读取,我们选择一个键来访问。Lock() 互斥体以确保对状态的独占访问,读取所选键的值, Unlock() 互斥体,并增加 readOps 计数。

我们还将启动 10 个 goroutine 来模拟写入,使用与读取相同的锁模式。

让 10 个 goroutine 在状态和互斥体上工作一秒钟。采集和报告最终操作计数。手机和报告最终操作计数。用最后的锁状态,显示它是如何结束了。

package main

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

func main(){
    var state = make(map[int]int)

    var mutex = &sync.Mutex{}

    var readOps  uint64 = 0
    var writeOps uint64 = 0

    for r := 0 ; r < 100 ; r ++{
        go func() {
            total := 0
            for{
                key:= rand.Intn(5)
                mutex.Lock()
                total += state[key]
                mutex.Unlock()
                atomic.AddUint64(&readOps , 1)

                time.Sleep(time.Millisecond)
            }
        }()
    }

    for w:= 0 ; w < 10 ; w ++{
        go func() {
            for{
                key := rand.Intn(5)
                val := rand.Intn(100)
                mutex.Lock()
                state[key] = val
                mutex.Unlock()
                atomic.AddUint64(&writeOps,1)
                time.Sleep(time.Millisecond)
            }
        }()
    }

    time.Sleep(time.Second)

    readOpsFinal := atomic.LoadUint64(&readOps)
    fmt.Println("readops:",readOpsFinal)

    writeOpsFinal := atomic.LoadUint64(&writeOps)
    fmt.Println("writeops",writeOpsFinal)

    mutex.Lock()
    fmt.Println("state",state)
    mutex.Unlock()
}

Go排序实例

Go语言的 sort 包实现了内置和用户定义类型的排序。排序方法特定于内置类型,排序是一个就地排序,不会生成新的切片。

这里举例了一个 int 一个 string 类型的例子。

package main

import (
    "sort"
    "fmt"
)

func main(){
    // example for string
    strs := []string{"c","a","b"}
    sort.Strings(strs)

    fmt.Println("Strings:",strs)

    //example of sorting
    ints := []int{7,2,4}
    sort.Ints(ints)
    fmt.Println("Ints: ",ints)

    //使用 sort 检查是相关 slice 是否已经被排好序
    s := sort.IntsAreSorted(ints)
    fmt.Println("sorted: ",s)
    
}

Go自定义函数

有时候,我们希望通过使用自然排序意外的其他方式对集合进行排序。例如,假设我们想通过字符串长度进行排序,下面是Go语言中自定义排序的类型。

为了使用Go语言中的自定义函数进行排序,我们需要一个相应的类型。这里创建了一个 ByLength 类型,它只是内置 []string 类型的别名。

需要实现 sort.Interface - Len , Less , Swap 在这个类型上,所以可以使用 sort 包中的一般 Sort 函数。 Len 和 Swap 通常在类型之间是相似的, Less 保存实际的自定义排序逻辑。在这个例子中,要按照字符串长度的增加顺序排序,因此在这里使用 len(s[i]) 和 len(s[j])。

所有这些都到位后,现在可以通过将原始 fruits 切片转换为 ByLength 来实现自定义排序,然后对该类型切片使用 sort.Sort() 方法。

package main

import (
    "sort"
    "fmt"
)

//定义类型
type ByLength []string

//重写下面几个接口函数
func (s ByLength) Len()int{
    return len(s)
}

func (s ByLength) Swap(i,j int){
    s[i],s[j] = s[j],s[i]
}

func (s ByLength) Less(i,j int) bool {
    return len(s[i]) < len(s[j])
}


//正常调用方法
func main(){
    fruits := []string{"peach","banana","kiwi"}
    sort.Sort(ByLength(fruits))
    fmt.Println(fruits)
}

Go panic 错误处理实例

Panic 通常意味着失去出乎意料的错了。大多数情况下,使用他来正常运行时不应该出现的错误,或者不准备妥善处理。

在程序中可使用 Panic 来检查意外的错误。

如果一个函数返回一个我们不知道如何(或想要)处理的错误值,那么 Panic 的常见用法就是终止。这里有一个例子,如果在创建一个新文件时需要意外的错误。

运行此程序将导致程序出错,打印错误消息和 goroutine 跟踪,并退出非零状态。

package main

import "os"

func main(){

    panic("a problem")

    _,err := os.Create("./test.log")
    if err != nil{
        panic(err)
    }
}

你可能感兴趣的:(Go 语言编程实例(四))