为了保护对象被并发访问修改,我们可以使用协程在后台顺序执行匿名函数来替代使用同步互斥锁。在下面的程序中我们有一个类型 Person
中包含一个字段 chF
,这是一个用于存放匿名函数的通道。
这个结构在构造函数 NewPerson()
中初始化的同时会启动一个后台协程 backend()
。backend()
方法会在一个无限循环中执行 chF
中放置的所有函数,有效地将它们序列化从而提供了安全的并发访问。更改和读取 salary
的方法会通过将一个匿名函数写入 chF
通道中,然后让 backend()
按顺序执行以达到其目的。需注意的是 Salary()
方法创建的闭包函数是如何将 fChan
通道包含在其中的。
当然,这是一个简化的例子,它不应该被用在这种案例下。但是它却向我们展示了在更复杂的场景中该如何解决这种问题。
示例:14.19-conc_access.go
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48
| package main
import ( "fmt" "strconv" )
type Person struct { Name string salary float64 chF chan func() }
func NewPerson(name string, salary float64) *Person { p := &Person{name, salary, make(chan func())} go p.backend() return p }
func (p *Person) backend() { for f := range p.chF { f() } }
func (p *Person) SetSalary(sal float64) { p.chF <- func() { p.salary = sal } }
func (p *Person) Salary() float64 { fChan := make(chan float64) p.chF <- func() { fChan <- p.salary } return <-fChan }
func (p *Person) String() string { return "Person - name is: " + p.Name + " - salary is: " + strconv.FormatFloat(p.Salary(), 'f', 2, 64) }
func main() { bs := NewPerson("Smith Bill", 2500.5) fmt.Println(bs) bs.SetSalary(4000.25) fmt.Println("Salary changed:") fmt.Println(bs) }
|
输出:
1 2 3
| Person - name is: Smith Bill - salary is: 2500.50 Salary changed: Person - name is: Smith Bill - salary is: 4000.25
|
链接