14.17 使用通道并发访问对象

为了保护对象被并发访问修改,我们可以使用协程在后台顺序执行匿名函数来替代使用同步互斥锁。在下面的程序中我们有一个类型 Person 中包含一个字段 chF ,这是一个用于存放匿名函数的通道。

这个结构在构造函数 NewPerson() 中初始化的同时会启动一个后台协程 backend()backend() 方法会在一个无限循环中执行 chF 中放置的所有函数,有效地将它们序列化从而提供了安全的并发访问。更改和读取 salary 的方法会通过将一个匿名函数写入 chF 通道中,然后让 backend() 按顺序执行以达到其目的。需注意的是 Salary() 方法创建的闭包函数是如何将 fChan 通道包含在其中的。

当然,这是一个简化的例子,它不应该被用在这种案例下。但是它却向我们展示了在更复杂的场景中该如何解决这种问题。

示例:14.19-conc_access.go

  1. package main
  2. import (
  3. "fmt"
  4. "strconv"
  5. )
  6. type Person struct {
  7. Name string
  8. salary float64
  9. chF chan func()
  10. }
  11. func NewPerson(name string, salary float64) *Person {
  12. p := &Person{name, salary, make(chan func())}
  13. go p.backend()
  14. return p
  15. }
  16. func (p *Person) backend() {
  17. for f := range p.chF {
  18. f()
  19. }
  20. }
  21. // Set salary.
  22. func (p *Person) SetSalary(sal float64) {
  23. p.chF <- func() { p.salary = sal }
  24. }
  25. // Retrieve salary.
  26. func (p *Person) Salary() float64 {
  27. fChan := make(chan float64)
  28. p.chF <- func() { fChan <- p.salary }
  29. return <-fChan
  30. }
  31. func (p *Person) String() string {
  32. return "Person - name is: " + p.Name + " - salary is: " + strconv.FormatFloat(p.Salary(), 'f', 2, 64)
  33. }
  34. func main() {
  35. bs := NewPerson("Smith Bill", 2500.5)
  36. fmt.Println(bs)
  37. bs.SetSalary(4000.25)
  38. fmt.Println("Salary changed:")
  39. fmt.Println(bs)
  40. }

输出:

  1. Person - name is: Smith Bill - salary is: 2500.50
  2. Salary changed:
  3. Person - name is: Smith Bill - salary is: 4000.25

链接