缓存复杂数据的最佳方法

时间:2018-12-15 10:47:30

标签: dictionary go

我有一个表,用于根据号码的前缀存储语音呼叫的费用:

Prefix  ratio 
44      0.01597
447     0.04958
447530  0.03
447531  0.048
447532  0.04950
1       0.1
97      0.1

在表中查找数字前缀并不复杂,因为需要最大匹配前缀。 例如
4475122112的前缀是447
并且4475302112的前缀是447530

我想将表缓存在内存中,以通过减少数据库交互来提高性能。 由于要获取数字前缀(然后是其速率),因此需要在缓存上进行搜索

我发现了两种方法:

  1. 将其存储在纯地图中。在地图上搜索可以很容易,因为扫描了所有地图(可能很懒)。
  2. 将链接列表结构创建为树。而短前缀靠近根,最长前缀靠近叶子。

现在,缓存此类数据的最佳方法是什么?还是有其他机制?

1 个答案:

答案 0 :(得分:6)

将其存储在地图中,然后尝试寻找所需费用的数字。如果数字(键)不在地图中,请截断其最后一位并重复。这样,如果找到匹配项,请确保将是最长的前缀。

这是一个示例查找功能:

var prefixCostMap = map[uint64]float64{
    44:     0.01597,
    447:    0.04958,
    447530: 0.03,
    447531: 0.048,
    447532: 0.04950,
    1:      0.1,
    97:     0.1,
}

func lookup(num uint64) (longestPrefix uint64, cost float64, ok bool) {
    longestPrefix = num

    for longestPrefix > 0 {
        cost, ok = prefixCostMap[longestPrefix]
        if ok {
            break
        }
        longestPrefix = longestPrefix / 10 // Cut off last digit
    }

    return
}

测试:

fmt.Println(lookup(4475122112))
fmt.Println(lookup(4475302112))
fmt.Println(lookup(999))

输出(在Go Playground上尝试):

447 0.04958 true
447530 0.03 true
0 0 false

注意:这不支持以0开头的数字。如果您还需要处理数字,则可以将数字存储为字符串值,这样会保留开头的0数字。

string版本的外观如下:

var prefixCostMap = map[string]float64{
    "44":     0.01597,
    "447":    0.04958,
    "447530": 0.03,
    "447531": 0.048,
    "447532": 0.04950,
    "1":      0.1,
    "97":     0.1,
    "0123":   0.05,
}

func lookup(num string) (longestPrefix string, cost float64, ok bool) {
    longestPrefix = num

    for longestPrefix != "" {
        cost, ok = prefixCostMap[longestPrefix]
        if ok {
            break
        }
        longestPrefix = longestPrefix[:len(longestPrefix)-1] // Cut off last digit
    }

    return
}

测试:

fmt.Println(lookup("4475122112"))
fmt.Println(lookup("4475302112"))
fmt.Println(lookup("999"))
fmt.Println(lookup("0123456"))

输出(在Go Playground上尝试):

447 0.04958 true
447530 0.03 true
 0 false
0123 0.05 true