go - 如何从接口中提取数据
问题描述
我正在使用 Redigo 从 Redis 检索指标。虽然该包有一些内置的帮助程序来解析某些格式的响应中的数据,但我有一个我需要解析的格式,它没有被涵盖。
该命令的结果是接口类型,包含一个长度不确定的数组,每个数组中包含一组数据。如果我将结果作为字符串打印出来,我会得到(添加换行符以提高可读性)......
[
[name cgroup1 consumers %!s(int64=2) pending %!s(int64=90) last-delivered-id 1582572156729-0]
[name cgroup2 consumers %!s(int64=2) pending %!s(int64=110) last-delivered-id 1582572156729-0]
]
我将如何在 Go 中迭代这样的响应并访问数据。我发现了很多处理 JSON 和解组的示例,但没有发现与上述类似的示例。
使用的redis命令是...
xinfo groups <stream name>
Redis 中的数据,看起来是这样的……
1) 1) "name"
2) "cgroup1"
3) "consumers"
4) (integer) 2
5) "pending"
6) (integer) 90
7) "last-delivered-id"
8) "1582572156729-0"
2) 1) "name"
2) "cgroup2"
3) "consumers"
4) (integer) 2
5) "pending"
6) (integer) 110
7) "last-delivered-id"
8) "1582572156729-0"
如果我打印出它显示的类型......
[]interface {}
另一个问题是如何提取嵌套更深的信息。
例如
XINFO STREAM <stream name>
我可以使用 ScanStruct 从第一级获取所有内容,但似乎无法获取第一个条目下的信息。
我的信息结构
type Stream struct {
Length int `redis:"length"`
Groups int `redis:"groups"`
LastID string `redis:"last-generated-id"`
}
如果我尝试从下面添加第一个条目,我不确定要使用什么类型或如何获取值 12[1] (1582572131616-0)
示例 Redis 输出
127.0.0.1:6379> xinfo stream stream1
1) "length"
2) (integer) 1200
3) "radix-tree-keys"
4) (integer) 12
5) "radix-tree-nodes"
6) (integer) 30
7) "groups"
8) (integer) 2
9) "last-generated-id"
10) "1582642828055-10"
11) "first-entry"
12) 1) "1582572131616-0"
2) 1) "payload"
2) "message:0"
13) "last-entry"
14) 1) "1582642828055-10"
2) 1) "payload"
2) "message:99"
解决方案
使用redis.Values转换interface{}
为切片。使用redis.ScanStruct从名称-值对的切片中设置结构字段。
type Group struct {
Name string `redis:"name"`
Consumers int `redis:"consumers"`
Pending int `redis:"pending"`
LastDeliveredID string `redis:"last-delivered-id"`
}
func scanGroups(resp interface{}, err error) ([]*Group, error) {
// Get slice from resp
groups, err := redis.Values(resp, err)
if err != nil {
return nil, err
}
var result []*Group
// For each group
for _, g := range groups {
// Get slice for group
v, err := redis.Values(g, nil)
if err != nil {
return nil, err
}
// Scan slice to struct
var group Group
err = redis.ScanStruct(v, &group)
if err != nil {
return nil, err
}
// Accumlate result
result = append(result, &group)
}
return result, nil
}
像这样使用它:
groups, err := scanGroups(conn.Do("XINFO", "GROUPS", "mystream"))
推荐阅读
- javascript - SetInterval 设置为 10ms 比预期的要多
- arrays - 将目录中的文件放入数组变量中
- python - 选项参数的标准 cmd 行分隔
- c# - C#,低功耗。为什么 GetGattServicesAsync 永远挂起?是否有解决方法,例如关闭/打开 BLE?
- c++11 - 我尝试使用 ascii 字符制作游戏,但它闪烁太多
- python - 带有字典python的熊猫数据框中的StopIteration问题
- javascript - 可拖动元素变为空
- swift - 有没有办法将 PDFTable 分解到不同的页面上?
- python - 为什么我们每次都需要重新创建模型?
- javascript - 寻找用于简单表格编辑的 localStorage 解决方案(包括页面 HTML)