乔克视界 乔克视界
首页
  • 运维
  • 开发
  • 监控
  • 安全
  • 随笔
  • Docker
  • Golang
  • Python
  • AIOps
  • 心情杂货
  • 读书笔记
  • 面试
  • 实用技巧
  • 博客搭建
友链
关于
收藏
  • 分类
  • 标签
  • 归档

乔克

云原生爱好者
首页
  • 运维
  • 开发
  • 监控
  • 安全
  • 随笔
  • Docker
  • Golang
  • Python
  • AIOps
  • 心情杂货
  • 读书笔记
  • 面试
  • 实用技巧
  • 博客搭建
友链
关于
收藏
  • 分类
  • 标签
  • 归档
  • Docker

  • Golang

    • Golang基础知识

    • Golang进阶知识

    • Golang常用包

      • Cobra 使用
      • gorm的简单操作
      • Excelize操作excl
      • sjson为json数据设置值
        • 支持的类型
        • 修改数组
        • 删除
        • 错误处理
          • 非法 JSON 串
          • 非法键路径
      • gjson解析json数据
      • yaml配置文件解析
      • viper读取配置文件
      • zap日志管理
      • gopsutil获取系统信息
  • AIOps

  • 专栏
  • Golang
  • Golang常用包
乔克
2025-07-17
目录

sjson为json数据设置值

安装:

$ go get github.com/tidwall/sjson
1

使用:

package main
import (
  "fmt"
  "github.com/tidwall/sjson"
)
const json = `{"name":{"first":"li","last":"dj"},"age":18}`
func main() {
  value, _ := sjson.Set(json, "name.last", "dajun")
  fmt.Println(value)
}
1
2
3
4
5
6
7
8
9
10

上面代码通过sjson.Set()将 JSON 串中name.last对应的值设置为dajun。与gjson一样,sjson也通过键路径指定具体的位置,键路径即为一系列以.分隔的键。sjson支持的键路径语法是gjson的一个子集,具体键路径的语法可以参见上一篇文章。sjson.Set()返回设置之后的 JSON 串。最终程序输出:

{"name":{"first":"li","last":"dajun"},"age":18}
1

# 支持的类型

sjson支持的类型包括nil/bool/int/float/string等。如果传入sjson不支持的类型,sjson会调用json.Marshal,然后将生成的字符串设置到对应的键路径上:

type User struct {
  Name string `json:"name"`
  Age  int    `json:"age"`
}
func main() {
  nilJSON, _ := sjson.Set("", "key", nil)
  fmt.Println(nilJSON)
  boolJSON, _ := sjson.Set("", "key", false)
  fmt.Println(boolJSON)
  intJSON, _ := sjson.Set("", "key", 1)
  fmt.Println(intJSON)
  floatJSON, _ := sjson.Set("", "key", 10.5)
  fmt.Println(floatJSON)
  strJSON, _ := sjson.Set("", "key", "hello")
  fmt.Println(strJSON)
  mapJSON, _ := sjson.Set("", "key", map[string]interface{}{"hello": "world"})
  fmt.Println(mapJSON)
  u := User{Name: "dj", Age: 18}
  structJSON, _ := sjson.Set("", "key", u)
  fmt.Println(structJSON)
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21

注意,我们传入一个空字符串,sjson.Set()会生成一个空对象,然后按照键路径依次设置值。下面分析上述程序输出:

  • nil:在 JSON 中用null表示,输出{"key":null};
  • false:在 JSON 中布尔值用true/false表示,输出{"key":false};
  • 1和10.5:整数和浮点数在 JSON 中都用number表示,分别输出{"key":1}和{"key":10.5};
  • hello:输出{"key":"hello"};
  • map[string]interface{}:sjson并不原生支持map类型,故通过json.Marshal将其序列化为{"hello":"world"}再设置到键key上,输出{"key":{"hello":"world"}};
  • User对象:先通过json.Marshal序列化为{"name":"dj","age":18}再设置;

# 修改数组

修改数组可以通过在键路径后添加索引,有两种特殊情况:

  • 使用-1或数组长度为索引表示在数组后添加一个新元素;
  • 使用的索引超出数组的长度,会在数组中添加很多null值。

看下面示例:

func main() {
  fruits := `{"fruits":["apple", "orange", "banana"]}`
  var newValue string
  newValue, _ = sjson.Set(fruits, "fruits.1", "grape")
  fmt.Println(newValue)
  newValue, _ = sjson.Set(fruits, "fruits.3", "pear")
  fmt.Println(newValue)
  newValue, _ = sjson.Set(fruits, "fruits.-1", "strawberry")
  fmt.Println(newValue)
  newValue, _ = sjson.Set(fruits, "fruits.5", "watermelon")
  fmt.Println(newValue)
}
1
2
3
4
5
6
7
8
9
10
11
12
  • fruits.1:设置第二个水果为grape(索引从 0 开始),输出{"fruits":["apple", "grape", "banana"]};
  • fruits.3:由于数组长度为 3,使用 3 表示在数组后添加一个元素,输出{"fruits":["apple","orange","banana","pear"]};
  • fruits.-1:使用-1同样表示在数组后添加一个元素,输出{"fruits":["apple", "orange", "banana","strawberry"]};
  • fruits.5:索引 5 已经大于数组长度 3 了,所以会多出两个null,输出{"fruits":["apple","orange","banana",null,null,"watermelon"]}。

# 删除

删除数组元素需要调用sjson.Delete()方法,键路径语法相同。如果键路径对应的值不存在,则Delete()无效果:

func main() {
  var newValue string
  user := `{"name":{"first":"li","last":"dj"},"age":18}`
  newValue, _ = sjson.Delete(user, "name.first")
  fmt.Println(newValue)
  newValue, _ = sjson.Delete(user, "name.full")
  fmt.Println(newValue)
  fruits := `{"fruits":["apple", "orange", "banana"]}`
  newValue, _ = sjson.Delete(fruits, "fruits.1")
  fmt.Println(newValue)
  newValue, _ = sjson.Delete(fruits, "fruits.-1")
  fmt.Println(newValue)
  newValue, _ = sjson.Delete(fruits, "fruits.5")
  fmt.Println(newValue)
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
  • name.first:删除字段name.first,输出{"name":{"last":"dj"},"age":18};
  • name.full:由于字段name.full不存在,无效果,输出{"name":{"first":"li","last":"dj"},"age":18};
  • fruits.1:删除数组fruits的第二个元素,输出{"fruits":["apple", "banana"]};
  • fruits.-1:删除数组最后一个元素,输出{"fruits":["apple", "orange"]};
  • fruits.5:索引 5 超出数组长度 3,无效果,输出{"fruits":["apple", "orange", "banana"]}。

# 错误处理

使用sjson出现的错误分为两种,一种是传入的 JSON 串不是合法的串,另一种是键路径语法错误。Set()和Delete()方法返回的第二个参数为错误,只有非法的键路径会返回错误,非法 JSON 串不会。

# 非法 JSON 串

同gjson一样,sjson同样不会检查传入的 JSON 串的合法性,它假设传入的是合法的串。如果传入一个非法的 JSON 串,程序输出不确定的结果:

func main() {
  user := `{"name":dj,age:18}`
  newValue, err := sjson.Set(user, "name", "dajun")
  fmt.Println(err, newValue)
}
1
2
3
4
5

上面程序中,我故意传入一个非法的 JSON 串(dj和age漏掉了双引号)。最终程序输出:

<nil> {"name":dj,age:"dajun"}
1

将age变为了dajun,显然不正确。然而此时返回的err = nil。

# 非法键路径

与gjson相比,sjson能使用的键路径语法比较有限,不能使用通配符和一些条件语法。如果传入的键路径非法,将返回非空的错误值:

func main() {
  user := `{"name":"dj","age":18}`
  newValue, err := sjson.Set(user, "na?e", "dajun")
  fmt.Println(err, newValue)
}
1
2
3
4
5

上次使用通配符?,输出:

wildcard characters not allowed in path
1
上次更新: 2025/07/18, 11:55:50
Excelize操作excl
gjson解析json数据

← Excelize操作excl gjson解析json数据→

最近更新
01
2025年,SRE在企业中可以做哪些事
07-18
02
SRE 如何提升自己在团队中的影响力
07-18
03
使用Go开发MCP服务
07-18
更多文章>
Theme by Vdoing | Copyright © 2019-2025 乔克 | MIT License | 渝ICP备20002153号 |
  • 跟随系统
  • 浅色模式
  • 深色模式
  • 阅读模式