about blog github

11 May 2018
go技巧分享(二)

这是一个系列文章,主要分享go的使用建议和技巧,每次分享3点,希望你能有所收获。

1 并发访问map

package main

import "sync"

type Map struct {
    sync.RWMutex
    Data map[int]int
}

func main() {
    m := Map{}
    m.Data = make(map[int]int)

    for i := 0; i < 100000; i++ {
        go m.Write(i, i)
        go m.Read(i)
    }
}

func (m *Map) Read(key int) int {
    m.RLock()
    value := m.Data[key]
    m.RUnlock()
    return value
}

func (m *Map) Write(key int, value int) {
    m.Lock()
    m.Data[key] = value
    m.Unlock()
}

注释掉加锁和解锁的代码:

func (m *Map) Read(key int) int {
    // m.RLock()
    value := m.Data[key]
    // m.RUnlock()
    return value
}

func (m *Map) Write(key int, value int) {
    // m.Lock()
    m.Data[key] = value
    // m.Unlock()
}

/*
fatal error: concurrent map read and map write
或者
fatal error: concurrent map writes
*/

多协程并发访问map时,有可能会导致程序退出,错误信息为:fatal error: concurrent map read and map write 或者 fatal error: concurrent map writes,所以如果希望在多协程中并发访问map,必须提供某种同步机制,一般情况下通过读写锁sync.RWMutex实现对map的并发访问控制,将map和sync.RWMutex封装一下,可以实现对map的安全并发访问。示例中,如果注释掉加锁和解锁的代码,运行时就会出现并发访问map错误。

2 结构体转换成json字符串时忽略字段

package main

import (
    "encoding/json"
    "fmt"
)

type User struct {
    Name string
    Age  int
    Addr string `json:"-"`
    // addr string
}

func main() {
    jsonStr, _ := json.Marshal(User{"user1", 12, "addr"})
    fmt.Printf("%s\n", jsonStr)
}

/*
{"Name":"user1","Age":12}
*/

将结构体转换成json字符串时,如果想忽略某些字段,可以通过在该字段后面添加`json:”-“`实现,添加该tag后,转换成json字符串时会忽略该字段。当然,你也可以通过将该字段首字母改成小写实现,只是这样的话该字段对其他包也隐藏了,两种方法各有利弊,权衡使用。

3 判断2个结构体是否相等

package main

import (
    "fmt"
    "reflect"
)

type User struct {
    Name string
    Age  int
    Addr string
}

func NewUser(name string, age int) User {
    u := User{}
    u.Name = name
    u.Age = age
    return u
}

func main() {
    u1 := NewUser("user1", 12)
    u2 := NewUser("user1", 12)
    fmt.Println(reflect.DeepEqual(u1, u2))
    fmt.Println(reflect.DeepEqual([]int{1, 2}, []int{1, 2}))
    fmt.Println(reflect.DeepEqual([2]int{1, 2}, [2]int{1, 2}))
    fmt.Println(reflect.DeepEqual(map[int]int{1: 1, 2: 2}, map[int]int{1: 1, 2: 2}))
}

/*
true
true
true
true
*/

在使用go语言过程中,有时需要比较2个结构体是否“相等”,即结构体字段和字段值都相同。如果自己实现比较函数,当结构体字段多时比较麻烦。通过使用reflect.DeepEqual可以实现判断2个结构体是否相等,但会有一定的性能影响。另外,reflect.DeepEqual不但可以比较结构体,像切片、数组和字典等都可以比较。



LEo at 00:05

about blog github