go - 如何在自定义类型的切片上进行范围
问题描述
我正在尝试为 Google DataStore 编写 Go 自定义缓存(更准确地说 - 围绕现有缓存库之一的包装器)。在缓存初始化时,它应该接受任何自定义类型的结构(具有适当定义的数据存储字段),这将成为所有存储项目的基础。这个想法是可以为各种类型创建/初始化缓存,这些类型反映了特定 DataStore 条目(CustomEntry)的结构
方法 1 - 存储reflect.Type并使用它。遇到的问题- 无法迭代自定义类型的切片
type CustomEntry struct {
Data struct {
name string `datastore:"name,noindex"`
address []string `datastore:"address,noindex"`
} `datastore:"data,noindex"`
}
func (cache *MyCache) CacheData(dataQuery string, dataType reflect.Type) {
slice := reflect.MakeSlice(reflect.SliceOf(dataType), 10, 10)
if keys, err := DataStoreClient.GetAll(cache.ctx, datastore.NewQuery(dataQuery), &slice); err != nil {
//handle error
} else {
for i, dataEntry:= range slice {
// ERROR: Cannot range over 'slice' (type Value)
cache.Set(keys[i].Name, dataEntry)
}
}
//usage: Cache.CacheData("Person", reflect.TypeOf(CustomEntry{})
方法 2 - 接受一个接口数组作为参数。遇到的问题= []CustomEntry不是[]interface{}
func (cache *MyCache) CacheData(dataQuery string, dataType []interface{}) {
if keys, err := DataStoreClient.GetAll(cache.ctx, datastore.NewQuery(dataQuery), &dataType); err != nil {
//handle error
} else {
for i, dataEntry:= range slice {
// this seems to work fine
cache.Set(keys[i].Name, dataEntry)
}
}
//usage:
var dataType []CustomEntry
Cache.CacheData("Person", data)
// ERROR: Cannot use 'data' (type []CustomEntry) as type []interface{}
任何建议将不胜感激。
解决方案
我找到了一个解决方案,并认为它可能值得分享,以防其他人有类似的问题。
最简单的方法是初始化 DataStore 预期接收的结构切片,然后将指向它的指针作为参数 (interface{}) 传递给所需的函数。DataStore,类似于一些解组函数(我尝试过使用 JSON 包)将能够成功地将数据附加到它。
在给定特定类型的情况下,尝试在函数中动态创建切片,然后由函数(例如 DataStore 客户端)接受该切片可能非常困难(我还没有设法找到方法)。同样,传递一部分接口(以便于迭代)只会使事情复杂化。
其次,为了迭代数据(例如将其存储在缓存中),有必要:(1)检索接口的底层值(即指针本身) - 这可以通过使用来实现reflect.ValueOf(pointerInterface)
,(2)取消引用指针,以便我们可以访问底层的、可迭代的结构切片——这可以通过调用来完成,(3)使用方法.Elem()
迭代底层切片(即使底层类型是可迭代的,也不会接受接口)。.Index(i)
range
自然地,添加一些 switch-case 语句可能适合确保捕获任何错误而不是导致运行时恐慌。
因此,以下代码为上述问题提供了一个可行的解决方案:
主要:
var data []customEntry
c.CacheData("Person",&data)
以及函数本身:
func (cache *MyCache) CacheData(dataQuery string, data interface{}) error {
if keys, err := DataStoreClient.GetAll(cache.ctx, datastore.NewQuery(dataQuery), data); err != nil {
return err
} else {
s := reflect.ValueOf(data).Elem()
for i := 0; i < s.Len(); i++ {
cache.Set(keys[i].Name, s.Index(i), 1)
}
}
}
推荐阅读
- php - 教义迁移不会识别注释的更改(在生产中)
- c# - 在遍历循环时将新对象添加到字典中?
- spring - Spring Boot:找不到位于带有@Value注释的资源文件夹中的文件
- reactjs - 如何在同一行对齐 3 张卡片
- javascript - 反应原生 Tabbar 颜色更改不起作用
- .net - 将 Application Insights 信息检索到 .Net 应用程序
- typescript - 打字稿类型约束不一致
- assembly - 使用合并排序构建包含来自 s1 和 s2 的所有字符的有序字节字符串
- if-statement - 如果获取 var 两个值
- unit-testing - Moq 验证调用时不匹配