- r - 以节省内存的方式增长 data.frame
- ruby-on-rails - ruby/ruby on rails 内存泄漏检测
- android - 无法解析导入android.support.v7.app
- UNIX 域套接字与共享内存(映射文件)
我正在寻找一种明确的方法来按顺序遍历 Go
map
。
Golang spec陈述如下:
The iteration order over maps is not specified and is not guaranteed to be the same from one iteration to the next. If map entries that have not yet been reached are removed during iteration, the corresponding iteration values will not be produced. If map entries are created during iteration, that entry may be produced during the iteration or may be skipped. The choice may vary for each entry created and from one iteration to the next. If the map is nil, the number of iterations is 0.
我在 StackOverflow 和谷歌搜索上找到的都是(恕我直言)我不喜欢的解决方法。
是否有可靠的方法来遍历 map 并按插入顺序检索项目?
我找到的解决方案是:
在两个单独的 slice 中跟踪键和值:这听起来像“不要使用 map ”,失去了使用 map 的所有优势。
使用映射但跟踪不同 slice 中的键:这意味着数据重复可能导致数据未对齐并最终可能带来大量错误和痛苦的调试。
你有什么建议?
根据可能的重复标记进行编辑。
我的问题与提供的问题( this question 和 this one )之间存在细微差别,这两个问题都要求按照键的字典顺序循环遍历 map ;相反,我特别询问了:
Is there a solid way to iterate through a map and retrieve items in the order they've been inserted?
它不是字典序的,因此不同于 @gramme.ninja
question :
How can I get the keys to be in order / sort the map so that the keys are in order and the values correspond?
最佳答案
如果您需要 map
和按顺序排列的键,那是两种不同的东西,您需要两种不同的(数据)类型来提供该功能。
实现此目的的最简单方法是在不同的 slice 中维护键顺序。每当您将新对放入 map 时,首先检查 key 是否已经在其中。如果不是,则将新 key 添加到单独的 slice 中。当你需要按顺序排列元素时,你可以使用 keys slice 。当然,当你删除一对时,你也必须从 slice 中删除它。
键 slice 只需要包含键(而不是值),因此开销很小。
将这个新功能(map+keys slice)包装成一个新类型并为其提供方法,并隐藏map和slice。这样就不会发生数据错位。
示例实现:
type Key int // Key type
type Value int // Value type
type Map struct {
m map[Key]Value
keys []Key
}
func New() *Map {
return &Map{m: make(map[Key]Value)}
}
func (m *Map) Set(k Key, v Value) {
if _, ok := m.m[k]; !ok {
m.keys = append(m.keys, k)
}
m.m[k] = v
}
func (m *Map) Range() {
for _, k := range m.keys {
fmt.Println(m.m[k])
}
}
使用它:
m := New()
m.Set(1, 11)
m.Set(2, 22)
m.Range()
在 Go Playground 上试试.
另一种方法是包装值,并且——沿着真实值——也存储下一个/上一个键。
例如,假设您想要一个类似 map[Key]Value
的 map :
type valueWrapper struct {
value Value
next *Key // Next key
}
无论何时向 map 添加一对,都将 valueWrapper
设置为值,并且必须将其链接到前一个(最后一个)对。要链接,您必须将最后一个包装器的next
字段设置为指向这个新键。为了轻松实现这一点,建议还存储最后一个 key (以避免必须搜索它)。
当你想按插入顺序遍历元素时,你从第一个开始(你必须存储它),它关联的 valueWrapper
会告诉你下一个键(按插入顺序) .
示例实现:
type Key int // Key type
type Value int // Value type
type valueWrapper struct {
v Value
next *Key
}
type Map struct {
m map[Key]valueWrapper
first, last *Key
}
func New() *Map {
return &Map{m: make(map[Key]valueWrapper)}
}
func (m *Map) Set(k Key, v Value) {
if _, ok := m.m[k]; !ok && m.last != nil {
w2 := m.m[*m.last]
m.m[*m.last] = valueWrapper{w2.v, &k}
}
w := valueWrapper{v: v}
m.m[k] = w
if m.first == nil {
m.first = &k
}
m.last = &k
}
func (m *Map) Range() {
for k := m.first; k != nil; {
w := m.m[*k]
fmt.Println(w.v)
k = w.next
}
}
使用是一样的。在 Go Playground 上试用.
注意:您可以根据自己的喜好改变一些东西:
你可以像 m map[Key]*valueWrapper
这样声明内部映射,所以在 Set()
中你可以改变 next
字段,而无需分配新的 valueWrapper
。
您可以选择 first
和 last
字段为 *valueWrapper
您可以选择 next
类型为 *valueWrapper
带有附加 slice 的方法更简单、更清晰。但是,如果映射变大,从中删除元素可能会变慢,因为我们还必须在 slice 中找到“未排序”的键,因此它的复杂度为 O(n)
。
如果您还向 valueWrapper< 中添加
结构。因此,如果您需要删除一个元素,您可以超快地找到包装器 (prev
字段,即使 map 很大,也可以轻松扩展 value-wrapper 中使用链表的方法以支持快速删除元素O(1)
),更新 prev 和 next 包装器(指向彼此),并执行一个简单的 delete()
操作,O(1)
。
请注意,第一个解决方案(带 slice )中的删除仍然可以通过使用一个额外的映射来加速,该映射将从键映射到 slice 中键的索引 (map[Key]int
), 所以删除操作仍然可以在O(1)
中实现,换取更大的复杂度。另一个加快速度的选择可能是将映射中的值更改为包装器,它可以保存 slice 中键的实际值和索引。
关于loops - 按顺序范围循环映射,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/39450120/
我不能解决这个问题。和标题说的差不多…… 如果其他两个范围/列中有“否”,我如何获得范围或列的平均值? 换句话说,我想计算 A 列的平均值,并且我有两列询问是/否问题(B 列和 C 列)。我只希望 B
我知道 python 2to3 将所有 xrange 更改为 range 我没有发现任何问题。我的问题是关于它如何将 range(...) 更改为 list(range(...)) :它是愚蠢的,只是
我有一个 Primefaces JSF 项目,并且我的 Bean 注释有以下内容: @Named("reportTabBean") @SessionScoped public class Report
在 rails3 中,我在模型中制作了相同的范围。例如 class Common ?" , at) } end 我想将公共(public)范围拆分为 lib 中的模块。所以我试试这个。 module
我需要在另一个 View 范围 bean 中使用保存在 View 范围 bean 中的一些数据。 @ManagedBean @ViewScoped public class Attivita impl
为什么下面的代码输出4?谁能给我推荐一篇好文章来深入学习 javascript 范围。 这段代码返回4,但我不明白为什么? (function f(){ return f(); functio
我有一个与此结构类似的脚本 $(function(){ var someVariable; function doSomething(){ //here } $('#som
我刚刚开始学习 Jquery,但这些示例对我帮助不大...... 现在,以下代码发生的情况是,我有 4 个表单,我使用每个表单的链接在它们之间进行切换。但我不知道如何在第一个函数中获取变量“postO
为什么当我这样做时: function Dog(){ this.firstName = 'scrappy'; } Dog.firstName 未定义? 但是我可以这样做: Dog.firstNa
我想打印文本文件 text.txt 的选定部分,其中包含: tickme 1.1(no.3) lesson1-bases lesson2-advancedfurther para:using the
我正在编写一些 JavaScript 代码。我对这个关键字有点困惑。如何在 dataReceivedHandler 函数中访问 logger 变量? MyClass: { logger: nu
我有这个代码: Public Sub test() Dim Tgt As Range Set Tgt = Range("A1") End Sub 我想更改当前为“A1”的 Tgt 的引
我正忙于此工作,以为我会把它放在我们那里。 该数字必须是最多3个单位和最多5个小数位的数字,等等。 有效的 999.99999 99.9 9 0.99999 0 无效的 -0.1 999.123456
覆盖代码时: @Override public void open(ExecutionContext executionContext) { super.open(executio
我想使用 preg_match 来匹配数字 1 - 21。我如何使用 preg_match 来做到这一点?如果数字大于 21,我不想匹配任何东西。 example preg_match('([0-9]
根据docs range函数有四种形式: (range) 0 - 无穷大 (range end) 0 - 结束 (range start end)开始 - 结束 (range start end st
我知道有一个UISlider,但是有人已经制作了RangeSlider(用两个拇指吗?)或者知道如何扩展 uislider? 最佳答案 我认为你不能直接扩展 UISlider,你可能需要扩展 UICo
我正在尝试将范围转换为列表。 nums = [] for x in range (9000, 9004): nums.append(x) print nums 输出 [9000] [9
请注意:此问题是由于在运行我的修饰方法时使用了GraphQL解析器。这意味着this的范围为undefined。但是,该问题的基础知识对于装饰者遇到问题的任何人都是有用的。 这是我想使用的基本装饰器(
我正在尝试创建一个工具来从网页上抓取信息(是的,我有权限)。 到目前为止,我一直在使用 Node.js 结合 requests 和 Cheerio 来拉取页面,然后根据 CSS 选择器查找信息。我已经
我是一名优秀的程序员,十分优秀!