使用 Golang、redis 和 time 进行测试
来源:stackoverflow
时间:2024-04-13 08:18:34 420浏览 收藏
学习Golang要努力,但是不要急!今天的这篇文章《使用 Golang、redis 和 time 进行测试》将会介绍到等等知识点,如果你想深入学习Golang,可以关注我!我会持续更新相关文章的,希望对大家都能有所帮助!
我第一次尝试使用 redis 进行一些测试,但我对 hget/hset/hgetall 遇到了一些困惑。我的主要问题是我需要存储时间,并且我想使用哈希,因为我将不断更新时间。
首先,我读到了这样的 marshalbinary 函数如何拯救我:
func (f foo) marshalbinary() ([]byte, error) {
return json.marshal(f)
}
它的作用是将结构保存为 json 字符串,但只是作为字符串而不是实际的 redis 哈希。我最终所做的是一段相当大的样板代码,它使我想要将我的结构保存到映射中,并且该结构在 redis 中正确存储为哈希。
type foo struct {
number int `json:"number"`
atime time.time `json:"atime"`
string string `json:"astring"`
}
func (f foo) toredis() map[string]interface{} {
res := make(map[string]interface{})
rt := reflect.typeof(f)
rv := reflect.valueof(f)
if rt.kind() == reflect.ptr {
rt = rt.elem()
rv = rv.elem()
}
for i := 0; i < rt.numfield(); i++ {
f := rt.field(i)
v := rv.field(i)
switch t := v.interface().(type) {
case time.time:
res[f.tag.get("json")] = t.format(time.rfc3339)
default:
res[f.tag.get("json")] = t
}
}
return res
}
然后,在调用 hgetall(..).result() 时解析回我的 foo 结构,我将结果作为 map[string]string 并使用以下函数创建一个新的 foo:
func setRequestParam(arg *Foo, i int, value interface{}) {
v := reflect.ValueOf(arg).Elem()
f := v.Field(i)
if f.IsValid() {
if f.CanSet() {
if f.Kind() == reflect.String {
f.SetString(value.(string))
return
} else if f.Kind() == reflect.Int {
f.Set(reflect.ValueOf(value))
return
} else if f.Kind() == reflect.Struct {
f.Set(reflect.ValueOf(value))
}
}
}
}
func fromRedis(data map[string]string) (f Foo) {
rt := reflect.TypeOf(f)
rv := reflect.ValueOf(f)
for i := 0; i < rt.NumField(); i++ {
field := rt.Field(i)
v := rv.Field(i)
switch v.Interface().(type) {
case time.Time:
if val, ok := data[field.Tag.Get("json")]; ok {
if ti, err := time.Parse(time.RFC3339, val); err == nil {
setRequestParam(&f, i, ti)
}
}
case int:
if val, ok := data[field.Tag.Get("json")]; ok {
in, _ := strconv.ParseInt(val, 10, 32)
setRequestParam(&f, i, int(in))
}
default:
if val, ok := data[field.Tag.Get("json")]; ok {
setRequestParam(&f, i, val)
}
}
}
return
}
整个代码的不光彩都在这里
我在想一定有更明智的方法来解决这个问题吗?或者我被迫做这样的事情?我需要存储的结构仅包含整数、字符串和时间。
*编辑 评论字段有点短,因此请进行编辑:
我最初确实像评论中建议的“傻瓜”和答案一样解决了这个问题。我之所以更改为上述部分,虽然解决方案更复杂,但我认为它对于更改来说更稳健。如果我采用硬编码地图解决方案,我“必须”具有:
- 带有字段哈希键的常量,因为它们至少会在两个地方使用(从 redis 到 redis),所以这将是编译器无法发现的愚蠢错误的地方。当然可以跳过这个,但知道我自己的拼写,这很可能会发生
- 如果有人只是想添加一个新字段并且不太了解代码,那么它可以很好地编译,但新字段不会添加到 redis 中。这是一个很容易犯的错误,特别是对于有点天真的初级开发人员,或者过于自信的高级开发人员。
- 我可以将这些辅助函数放入库中,当需要时间或复杂类型时,一切都会神奇地适用于我们的所有代码。
我想要的问题/希望是:我真的必须像这样跳过障碍才能用 go 将时间存储在 redis 哈希中吗?公平,time.time 不是一个基元,redis 也不是一个(无)sql 数据库,但我认为缓存中的时间戳是一个非常常见的用例(在我的例子中,是一个心跳来跟踪超时会话和元数据)足以永久存储它,因此需要更新它们)。但也许我误用了 redis,我应该有两个条目,一个用于数据,一个用于时间戳,这样我就只剩下两个简单的 get/set 函数,接收 time.time 并返回 time.time。
正确答案
您可以使用 redigo/redis#Args.AddFlat 将结构转换为 redis 哈希,我们可以使用 redis 标签映射该值。
package main
import (
"fmt"
"time"
"github.com/gomodule/redigo/redis"
)
type foo struct {
number int64 `json:"number" redis:"number"`
atime time.time `json:"atime" redis:"atime"`
astring string `json:"astring" redis:"astring"`
}
func main() {
c, err := redis.dial("tcp", ":6379")
if err != nil {
fmt.println(err)
return
}
defer c.close()
t1 := time.now().utc()
var foo foo
foo.number = 10000000000
foo.atime = t1
foo.astring = "hello"
tmp := redis.args{}.add("id1").addflat(&foo)
if _, err := c.do("hmset", tmp...); err != nil {
fmt.println(err)
return
}
v, err := redis.stringmap(c.do("hgetall", "id1"))
if err != nil {
fmt.println(err)
return
}
fmt.printf("%#v\n", v)
}
然后要更新atime,您可以使用redis hset
if _, err := c.do("hmset", "id1", "atime", t1.add(-time.hour * (60 * 60 * 24))); err != nil {
fmt.println(err)
return
}
为了将其检索回结构,我们必须执行一些 reflect 魔法
func structfrommap(src map[string]string, dst interface{}) error {
dt := reflect.typeof(dst).elem()
dv := reflect.valueof(dst).elem()
for i := 0; i < dt.numfield(); i++ {
sf := dt.field(i)
sv := dv.field(i)
if v, ok := src[strings.tolower(sf.name)]; ok {
switch sv.interface().(type) {
case time.time:
format := "2006-01-02 15:04:05 -0700 mst"
ti, err := time.parse(format, v)
if err != nil {
return err
}
sv.set(reflect.valueof(ti))
case int, int64:
x, err := strconv.parseint(v, 10, sv.type().bits())
if err != nil {
return err
}
sv.setint(x)
default:
sv.setstring(v)
}
}
}
return nil
}
最终代码
package main
import (
"fmt"
"time"
"reflect"
"strings"
"strconv"
"github.com/gomodule/redigo/redis"
)
type Foo struct {
Number int64 `json:"number" redis:"number"`
ATime time.Time `json:"atime" redis:"atime"`
AString string `json:"astring" redis:"astring"`
}
func main() {
c, err := redis.Dial("tcp", ":6379")
if err != nil {
fmt.Println(err)
return
}
defer c.Close()
t1 := time.Now().UTC()
var foo Foo
foo.Number = 10000000000
foo.ATime = t1
foo.AString = "Hello"
tmp := redis.Args{}.Add("id1").AddFlat(&foo)
if _, err := c.Do("HMSET", tmp...); err != nil {
fmt.Println(err)
return
}
v, err := redis.StringMap(c.Do("HGETALL", "id1"))
if err != nil {
fmt.Println(err)
return
}
fmt.Printf("%#v\n", v)
if _, err := c.Do("HMSET", "id1", "atime", t1.Add(-time.Hour * (60 * 60 * 24))); err != nil {
fmt.Println(err)
return
}
var foo2 Foo
structFromMap(v, &foo2)
fmt.Printf("%#v\n", foo2)
}
func structFromMap(src map[string]string, dst interface{}) error {
dt := reflect.TypeOf(dst).Elem()
dv := reflect.ValueOf(dst).Elem()
for i := 0; i < dt.NumField(); i++ {
sf := dt.Field(i)
sv := dv.Field(i)
if v, ok := src[strings.ToLower(sf.Name)]; ok {
switch sv.Interface().(type) {
case time.Time:
format := "2006-01-02 15:04:05 -0700 MST"
ti, err := time.Parse(format, v)
if err != nil {
return err
}
sv.Set(reflect.ValueOf(ti))
case int, int64:
x, err := strconv.ParseInt(v, 10, sv.Type().Bits())
if err != nil {
return err
}
sv.SetInt(x)
default:
sv.SetString(v)
}
}
}
return nil
}
注意:结构体字段名称与 redis 标记匹配
本篇关于《使用 Golang、redis 和 time 进行测试》的介绍就到此结束啦,但是学无止境,想要了解学习更多关于Golang的相关知识,请关注golang学习网公众号!
-
502 收藏
-
502 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
139 收藏
-
204 收藏
-
325 收藏
-
478 收藏
-
486 收藏
-
439 收藏
-
357 收藏
-
352 收藏
-
101 收藏
-
440 收藏
-
212 收藏
-
143 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 485次学习