Skip to content

Commit 903958d

Browse files
korzhaoBryan C. Mills
authored and
Bryan C. Mills
committed
encoding/gob: marshal maps using reflect.Value.MapRange
golang.org/cl/33572 added a map iterator. use the reflect.Value.MapRange to fix map keys that contain a NaN Fixes #24075 Change-Id: I0214d6f26c2041797703e48eac16404f189d6982 GitHub-Last-Rev: 5c01e11 GitHub-Pull-Request: #47476 Reviewed-on: https://go-review.googlesource.com/c/go/+/338609 Trust: Bryan C. Mills <bcmills@google.com> Run-TryBot: Bryan C. Mills <bcmills@google.com> TryBot-Result: Go Bot <gobot@golang.org> Reviewed-by: Rob Pike <r@golang.org>
1 parent 6640171 commit 903958d

File tree

2 files changed

+55
-5
lines changed

2 files changed

+55
-5
lines changed

src/encoding/gob/encode.go

Lines changed: 5 additions & 5 deletions
Original file line numberDiff line numberDiff line change
@@ -368,11 +368,11 @@ func (enc *Encoder) encodeMap(b *encBuffer, mv reflect.Value, keyOp, elemOp encO
368368
state := enc.newEncoderState(b)
369369
state.fieldnum = -1
370370
state.sendZero = true
371-
keys := mv.MapKeys()
372-
state.encodeUint(uint64(len(keys)))
373-
for _, key := range keys {
374-
encodeReflectValue(state, key, keyOp, keyIndir)
375-
encodeReflectValue(state, mv.MapIndex(key), elemOp, elemIndir)
371+
state.encodeUint(uint64(mv.Len()))
372+
mi := mv.MapRange()
373+
for mi.Next() {
374+
encodeReflectValue(state, mi.Key(), keyOp, keyIndir)
375+
encodeReflectValue(state, mi.Value(), elemOp, elemIndir)
376376
}
377377
enc.freeEncoderState(state)
378378
}

src/encoding/gob/encoder_test.go

Lines changed: 50 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,9 @@ import (
99
"encoding/hex"
1010
"fmt"
1111
"io"
12+
"math"
1213
"reflect"
14+
"sort"
1315
"strings"
1416
"testing"
1517
)
@@ -1152,3 +1154,51 @@ func TestDecodeErrorMultipleTypes(t *testing.T) {
11521154
t.Errorf("decode: expected duplicate type error, got %s", err.Error())
11531155
}
11541156
}
1157+
1158+
// Issue 24075
1159+
func TestMarshalFloatMap(t *testing.T) {
1160+
nan1 := math.NaN()
1161+
nan2 := math.Float64frombits(math.Float64bits(nan1) ^ 1) // A different NaN in the same class.
1162+
1163+
in := map[float64]string{
1164+
nan1: "a",
1165+
nan1: "b",
1166+
nan2: "c",
1167+
}
1168+
1169+
var b bytes.Buffer
1170+
enc := NewEncoder(&b)
1171+
if err := enc.Encode(in); err != nil {
1172+
t.Errorf("Encode : %v", err)
1173+
}
1174+
1175+
out := map[float64]string{}
1176+
dec := NewDecoder(&b)
1177+
if err := dec.Decode(&out); err != nil {
1178+
t.Fatalf("Decode : %v", err)
1179+
}
1180+
1181+
type mapEntry struct {
1182+
keyBits uint64
1183+
value string
1184+
}
1185+
readMap := func(m map[float64]string) (entries []mapEntry) {
1186+
for k, v := range m {
1187+
entries = append(entries, mapEntry{math.Float64bits(k), v})
1188+
}
1189+
sort.Slice(entries, func(i, j int) bool {
1190+
ei, ej := entries[i], entries[j]
1191+
if ei.keyBits != ej.keyBits {
1192+
return ei.keyBits < ej.keyBits
1193+
}
1194+
return ei.value < ej.value
1195+
})
1196+
return entries
1197+
}
1198+
1199+
got := readMap(out)
1200+
want := readMap(in)
1201+
if !reflect.DeepEqual(got, want) {
1202+
t.Fatalf("\nEncode: %v\nDecode: %v", want, got)
1203+
}
1204+
}

0 commit comments

Comments
 (0)