doujian4752 2018-08-16 20:46
浏览 407
已采纳

在sync.Map中加载或存储,而无需每次都创建新结构

Is it possible to LoadOrStore into a Go sync.Map without creating a new structure every time? If not, what alternatives are available?

The use case here is if I'm using the sync.Map as a cache where cache misses are rare (but possible) and on a cache miss I want to add to the map, I need to initialize a structure every single time LoadOrStore is called rather than just creating the struct when needed. I'm worried this will hurt the GC, initializing hundreds of thousands of structures that will not be needed.

In Java this can be done using computeIfAbsent.

  • 写回答

2条回答 默认 最新

  • dongxietao0263 2018-08-16 22:47
    关注

    Package sync

    import "sync"
    

    type Map

    Map is like a Go map[interface{}]interface{} but is safe for concurrent use by multiple goroutines without additional locking or coordination. Loads, stores, and deletes run in amortized constant time.

    The Map type is specialized. Most code should use a plain Go map instead, with separate locking or coordination, for better type safety and to make it easier to maintain other invariants along with the map content.

    The Map type is optimized for two common use cases: (1) when the entry for a given key is only ever written once but read many times, as in caches that only grow, or (2) when multiple goroutines read, write, and overwrite entries for disjoint sets of keys. In these two cases, use of a Map may significantly reduce lock contention compared to a Go map paired with a separate Mutex or RWMutex.


    The usual way to solve these problems is to construct a usage model and then benchmark it.

    For example, since "cache misses are rare", assume that Load wiil work most of the time and only LoadOrStore (with value allocation and initialization) when necessary.

    $ go test map_test.go -bench=. -benchmem
    BenchmarkHit-4     2     898810447 ns/op        44536 B/op        1198 allocs/op
    BenchmarkMiss-4    1    2958103053 ns/op    483957168 B/op    43713042 allocs/op
    $
    

    map_test.go:

    package main
    
    import (
        "strconv"
        "sync"
        "testing"
    )
    
    func BenchmarkHit(b *testing.B) {
        for N := 0; N < b.N; N++ {
            var m sync.Map
            for i := 0; i < 64*1024; i++ {
                for k := 0; k < 256; k++ {
    
                    // Assume cache hit
                    v, ok := m.Load(k)
                    if !ok {
                        // allocate and initialize value
                        v = strconv.Itoa(k)
                        a, loaded := m.LoadOrStore(k, v)
                        if loaded {
                            v = a
                        }
                    }
                    _ = v
    
                }
            }
        }
    }
    
    func BenchmarkMiss(b *testing.B) {
        for N := 0; N < b.N; N++ {
            var m sync.Map
            for i := 0; i < 64*1024; i++ {
                for k := 0; k < 256; k++ {
    
                    // Assume cache miss
                    // allocate and initialize value
                    var v interface{} = strconv.Itoa(k)
                    a, loaded := m.LoadOrStore(k, v)
                    if loaded {
                        v = a
                    }
                    _ = v
    
                }
            }
        }
    }
    
    本回答被题主选为最佳回答 , 对您是否有帮助呢?
    评论
查看更多回答(1条)

报告相同问题?

悬赏问题

  • ¥15 问题重发,R语言:代码运行过程中出现如下警告,请求如何解决!
  • ¥15 苍穹外卖拦截器token为null
  • ¥15 pyqt怎么把滑块和输入框相互绑定,求解决!
  • ¥20 wpf datagrid单元闪烁效果失灵
  • ¥15 券商软件上市公司信息获取问题
  • ¥100 ensp启动设备蓝屏,代码clock_watchdog_timeout
  • ¥15 Android studio AVD启动不了
  • ¥15 陆空双模式无人机怎么做
  • ¥15 想咨询点问题,与算法转换,负荷预测,数字孪生有关
  • ¥15 C#中的编译平台的区别影响