feature_reflect_array.go 2.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293
  1. package jsoniter
  2. import (
  3. "fmt"
  4. "io"
  5. "reflect"
  6. "unsafe"
  7. )
  8. func decoderOfArray(cfg *frozenConfig, prefix string, typ reflect.Type) ValDecoder {
  9. decoder := decoderOfType(cfg, prefix+"[array]->", typ.Elem())
  10. return &arrayDecoder{typ, typ.Elem(), decoder}
  11. }
  12. func encoderOfArray(cfg *frozenConfig, prefix string, typ reflect.Type) ValEncoder {
  13. encoder := encoderOfType(cfg, prefix+"[array]->", typ.Elem())
  14. if typ.Elem().Kind() == reflect.Map {
  15. encoder = &OptionalEncoder{encoder}
  16. }
  17. return &arrayEncoder{typ, typ.Elem(), encoder}
  18. }
  19. type arrayEncoder struct {
  20. arrayType reflect.Type
  21. elemType reflect.Type
  22. elemEncoder ValEncoder
  23. }
  24. func (encoder *arrayEncoder) Encode(ptr unsafe.Pointer, stream *Stream) {
  25. stream.WriteArrayStart()
  26. elemPtr := unsafe.Pointer(ptr)
  27. encoder.elemEncoder.Encode(elemPtr, stream)
  28. for i := 1; i < encoder.arrayType.Len(); i++ {
  29. stream.WriteMore()
  30. elemPtr = unsafe.Pointer(uintptr(elemPtr) + encoder.elemType.Size())
  31. encoder.elemEncoder.Encode(unsafe.Pointer(elemPtr), stream)
  32. }
  33. stream.WriteArrayEnd()
  34. if stream.Error != nil && stream.Error != io.EOF {
  35. stream.Error = fmt.Errorf("%v: %s", encoder.arrayType, stream.Error.Error())
  36. }
  37. }
  38. func (encoder *arrayEncoder) EncodeInterface(val interface{}, stream *Stream) {
  39. // special optimization for interface{}
  40. e := (*emptyInterface)(unsafe.Pointer(&val))
  41. if e.word == nil {
  42. stream.WriteArrayStart()
  43. stream.WriteNil()
  44. stream.WriteArrayEnd()
  45. return
  46. }
  47. elemType := encoder.arrayType.Elem()
  48. if encoder.arrayType.Len() == 1 && (elemType.Kind() == reflect.Ptr || elemType.Kind() == reflect.Map) {
  49. ptr := uintptr(e.word)
  50. e.word = unsafe.Pointer(&ptr)
  51. }
  52. if reflect.TypeOf(val).Kind() == reflect.Ptr {
  53. encoder.Encode(unsafe.Pointer(&e.word), stream)
  54. } else {
  55. encoder.Encode(e.word, stream)
  56. }
  57. }
  58. func (encoder *arrayEncoder) IsEmpty(ptr unsafe.Pointer) bool {
  59. return false
  60. }
  61. type arrayDecoder struct {
  62. arrayType reflect.Type
  63. elemType reflect.Type
  64. elemDecoder ValDecoder
  65. }
  66. func (decoder *arrayDecoder) Decode(ptr unsafe.Pointer, iter *Iterator) {
  67. decoder.doDecode(ptr, iter)
  68. if iter.Error != nil && iter.Error != io.EOF {
  69. iter.Error = fmt.Errorf("%v: %s", decoder.arrayType, iter.Error.Error())
  70. }
  71. }
  72. func (decoder *arrayDecoder) doDecode(ptr unsafe.Pointer, iter *Iterator) {
  73. offset := uintptr(0)
  74. iter.ReadArrayCB(func(iter *Iterator) bool {
  75. if offset < decoder.arrayType.Size() {
  76. decoder.elemDecoder.Decode(unsafe.Pointer(uintptr(ptr)+offset), iter)
  77. offset += decoder.elemType.Size()
  78. } else {
  79. iter.Skip()
  80. }
  81. return true
  82. })
  83. }