我有一个表,用于根据号码的前缀存储语音呼叫的费用:
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
我想将表缓存在内存中,以通过减少数据库交互来提高性能。 由于要获取数字前缀(然后是其速率),因此需要在缓存上进行搜索
我发现了两种方法:
现在,缓存此类数据的最佳方法是什么?还是有其他机制?
答案 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