diff --git a/feature_reflect_native.go b/feature_reflect_native.go index 0832101..4db4c96 100644 --- a/feature_reflect_native.go +++ b/feature_reflect_native.go @@ -5,6 +5,7 @@ import ( "encoding/base64" "encoding/json" "unsafe" + "reflect" ) type stringCodec struct { @@ -349,7 +350,12 @@ type emptyInterfaceCodec struct { } func (codec *emptyInterfaceCodec) Decode(ptr unsafe.Pointer, iter *Iterator) { - *((*interface{})(ptr)) = iter.Read() + existing := *((*interface{})(ptr)) + if existing != nil && reflect.TypeOf(existing).Kind() == reflect.Ptr { + iter.ReadVal(existing) + } else { + *((*interface{})(ptr)) = iter.Read() + } } func (codec *emptyInterfaceCodec) Encode(ptr unsafe.Pointer, stream *Stream) { diff --git a/jsoniter_interface_test.go b/jsoniter_interface_test.go index c037049..12cf39b 100644 --- a/jsoniter_interface_test.go +++ b/jsoniter_interface_test.go @@ -5,6 +5,7 @@ import ( "github.com/stretchr/testify/require" "testing" "unsafe" + "fmt" ) func Test_write_array_of_interface(t *testing.T) { @@ -297,3 +298,18 @@ func Test_array_with_nothing(t *testing.T) { should.Nil(err) should.Equal(`[null,null]`, output) } + +func Test_unmarshal_ptr_to_interface(t *testing.T) { + type TestData struct { + Name string `json:"name"` + } + should := require.New(t) + var obj interface{} = &TestData{} + err := json.Unmarshal([]byte(`{"name":"value"}`), &obj) + should.Nil(err) + should.Equal("&{value}", fmt.Sprintf("%v", obj)) + obj = interface{}(&TestData{}) + err = Unmarshal([]byte(`{"name":"value"}`), &obj) + should.Nil(err) + should.Equal("&{value}", fmt.Sprintf("%v", obj)) +}