golang map转json的顺序问题

本文深入探讨了在Go语言中,map数据结构转换为JSON格式时的内部处理机制。揭示了一个常见误解:尽管map本身无序,但在转换过程中,其键会按ASCII码升序排列,确保了输出的有序性和一致性。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

我们都知道map是无序的,每次取出key/value的顺序都可能不一致,但map转json的顺序是不是也是无序的吗?尽管json中的参数顺序大部分情况下对使用没有影响,我们不妨看看源码中怎么处理的。

先说结论:

map转json是有序的,按照ASCII码升序排列key。

溯源

源码位于encoding/json/encode.go中

type mapEncoder struct {
   elemEnc encoderFunc
}

func (me mapEncoder) encode(e *encodeState, v reflect.Value, opts encOpts) {
   if v.IsNil() {//为nil时,返回null
      e.WriteString("null")
      return
   }
   e.WriteByte('{')

   // Extract and sort the keys.
   keys := v.MapKeys()//获取map中的所有keys
   sv := make([]reflectWithString, len(keys))
   for i, v := range keys {
      sv[i].v = v
      if err := sv[i].resolve(); err != nil {//处理key,尤其是非string(int/uint)类型的key转string
         e.error(&MarshalerError{v.Type(), err})
      }
   }
   //排序,升序,直接比较字符串
   sort.Slice(sv, func(i, j int) bool { return sv[i].s < sv[j].s })

   for i, kv := range sv {
      if i > 0 {
         e.WriteByte(',')
      }
      e.string(kv.s, opts.escapeHTML)
      e.WriteByte(':')
      me.elemEnc(e, v.MapIndex(kv.v), opts)
   }
   e.WriteByte('}')
}

func newMapEncoder(t reflect.Type) encoderFunc {
   switch t.Key().Kind() {
   case reflect.String,
      reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64,
      reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr:
   default:
      if !t.Key().Implements(textMarshalerType) {
         return unsupportedTypeEncoder
      }
   }
   me := mapEncoder{typeEncoder(t.Elem())}
   return me.encode
}
Golang中,map换为JSON格式时,默认情况下是无序的。这是因为Golangmap是无序的数据结构,而JSON本身也是一个无序的键值对集合。所以,JSON的参数顺序在大多数情况下对使用没有影响。如果你希望生成有序的JSON数据,可以使用第三方库,例如`encoding/json`包中的`MarshalIndent`函数,通过设置`indent`参数来实现按照一定格式生成有序的JSON数据。你可以在代码中指定需要的键值对顺序,然后将map换为JSON数据。这样,生成的JSON数据将会按照你指定的顺序进行排列。 示例代码如下: ```go import ( "encoding/json" "fmt" "sort" ) func main() { m := map[string]interface{}{ "key1": "value1", "key3": "value3", "key2": "value2", } keys := make([]string, 0, len(m)) for k := range m { keys = append(keys, k) } sort.Strings(keys) orderedMap := make(map[string]interface{}) for _, k := range keys { orderedMap[k = m[k] } jsonData, err := json.MarshalIndent(orderedMap, "", " ") if err != nil { fmt.Println("Error:", err) return } fmt.Println(string(jsonData)) } ``` 以上示例代码中,我们先将map的键取出并排序,然后根据排序后的键重新构建一个有序的map。最后使用`json.MarshalIndent`函数将有序的map换为有序的JSON数据,并打印输出。这样就能生成按照键的顺序排列的JSON数据了。<span class="em">1</span><span class="em">2</span> #### 引用[.reference_title] - *1* [Golang map如何生成有序的json数据详解](https://download.csdn.net/download/weixin_38713039/12872560)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] - *2* [golang mapjson顺序问题](https://blog.csdn.net/xz_studying/article/details/103219478)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值