考虑这个基准,我们在这里比较地图访问与切换
var code = []int32{0, 10, 100, 100, 0, 10, 0, 10, 100, 14, 1000, 100, 1000, 0, 0, 10, 100, 1000, 10, 0, 1000, 12}
var mapCode = map[int32]int32{
0: 1,
10: 2,
100: 3,
1000: 4,
}
func BenchmarkMap(b *testing.B) {
success := int32(0)
fail := int32(0)
for n := 0; n < b.N; n++ {
// for each value in code array, do a specific action
for _, v := range code {
c, ok := mapCode[v]
if !ok {
fail++
} else {
success += c
}
}
}
}
func BenchmarkSwitch(b *testing.B) {
success := int32(0)
fail := int32(0)
for n := 0; n < b.N; n++ {
// for each value in code array, do a specific action
for _, v := range code {
switch v {
case 0:
success++
case 10:
success += 2
case 100:
success += 3
case 1000:
success += 4
default:
fail++
}
}
}
}
结果如下:
BenchmarkMap-2 5000000 277 ns/op 0 B/op 0 allocs/op
BenchmarkSwitch-2 30000000 48.2 ns/op 0 B/op 0 allocs/op
因此,使用地图似乎比切换要慢。
我目前正在尝试使用类似于的代码来优化功能BenchmarkMap()
,其中地图访问是瓶颈,但是由于程序启动时会动态生成地图,因此我无法使用switch(即,它可能会根据输入参数而变化)
有没有办法获得switch x {}
与动态生成的地图类似的性能?
映射不适用,因为索引索引是在运行时评估的,并且从映射中获取元素所涉及的操作比仅进行一次(切片)索引要多。某些switch
es(带有case
具有恒定表达式的分支)甚至可以在编译时进行优化。
但是地图并不是唯一的“动态”结构。另外一个,有切片。切片可以索引,就像地图一样。
是的,切片是基础数组 连续段 的描述符。这意味着,如果您的索引如1000
,则切片至少需要包含1000+1 = 1001
元素。
因此,如果您愿意为了性能而牺牲一些内存并使用切片而不是映射, 则甚至可以比使用以下switch
语句的解决方案更快:
var sliceCode = []int32{
0: 1,
10: 2,
100: 3,
1000: 4,
}
func BenchmarkSlice(b *testing.B) {
success := int32(0)
fail := int32(0)
for n := 0; n < b.N; n++ {
// for each value in code array, do a specific action
for _, v := range code {
c := sliceCode[v]
if c == 0 {
fail++
} else {
success += c
}
}
}
}
以及基准测试结果:
BenchmarkMap-4 10000000 148 ns/op
BenchmarkSlice-4 100000000 17.6 ns/op
BenchmarkSwitch-4 50000000 31.0 ns/op
在这个具体示例中,切片解决 方案的速度 比switch
解决方案快两倍!
笔记:
上面我提到过,如果您有一个类似的索引1000
,则至少需要1001
元素。这部分是正确的。例如,如果您有类似的索引990..1000
,则可以有一个简单的索引转换逻辑,例如index - 990
,那么仅包含11个元素的切片就足够了。
另请注意,在使用逗号分隔的习惯用法为地图编制索引时,我们能够判断元素是否在地图中。对于切片,我们没有该选项。因此,我们必须从有效的元素类型集中指定一个值,并将其用作“丢失”信号。在上面的示例中,0
对我们来说是完美的,因为它没有被使用(并且所有未明确列出的元素都0
默认设置为)。如果在您的示例中int32
可以使用所有有效值,则另一个选择是使用包装器或指针类型作为可能具有nil
值的切片的元素类型,指示缺少索引的元素。
问题内容: 我在JSF1.2和Richfaces 3.3.2中构建了一个树分页,因为我有很多树节点(大约80k),而且速度很慢。 因此,作为首次尝试,我将创建一个带有页面和页面节点列表的HashMap。 但是,性能还不够好… 所以我想知道是否是比HashMap更快的东西,也许是列表列表之类的东西。 有人对此有经验吗?我能做什么? 提前致谢。 编辑。 最大的问题是,我必须在树的子节点中验证用户的权限
本文向大家介绍arcgis android之地图图层切换,包括了arcgis android之地图图层切换的使用技巧和注意事项,需要的朋友参考一下 图层切换是arcgis for android 开发必常见的功能,我的实现方法很简单。 首先初始化多个图层服务对象,如下: 在onCreate()方法里创建一个默认图层: 最后当点击相应的图层按钮时做判断即可: 好了,今天就写到这里,以后我再介绍其他a
在地图主界面找到工具箱,打开工具箱,工具箱底部有切换地图的选项,目前支持三种地图分别为普通地图、黑夜地图、卫星地图,如上图所示。
问题内容: 可以将切片用作键吗? 我有尝试: 编译器给我一个错误。因此,要么不可能,要么我错误地声明了它(如果可以,那么正确的方法是什么?)。 问题答案: 不可以,切片没有定义相等性,因此无法用作映射键。
问题内容: 我有一个Map是一个到值列表的映射,但是我需要将其求逆,以便 变成 例如我有 我想看看 除了要遍历映射条目并创建set条目(如果不存在)并添加到列表等之外,Java 8中还有什么更简单的方法可以做到这一点?我一直认为这确实很明显,但我无法解决。 提前致谢 问题答案: 未经测试,但是您可以执行以下操作:
问题内容: 最近开始使用一个数据库,其中的约定是为每个表创建一个视图。如果您假设表和视图之间存在一对一的映射,那么我想知道是否有人可以告诉我这样做的性能影响。顺便说一句,这是在Oracle上。 问题答案: 假设问题是关于非具体化视图的,那么- 确实取决于该视图所基于的查询以及对其执行的操作。有时,谓词可以由优化器推入视图查询中。如果不是这样,那将不如表格本身好。视图建立在表格之上- 为什么您期望性