当前位置:首页 > 文章列表 > Golang > Go问答 > 使用 Golang、redis 和 time 进行测试

使用 Golang、redis 和 time 进行测试

来源:stackoverflow 2024-04-13 08:18:34 0浏览 收藏

学习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学习网公众号!

版本声明
本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
华为再出时尚耳机,FreeBuds Lipstick 2侧翻盖设计引关注华为再出时尚耳机,FreeBuds Lipstick 2侧翻盖设计引关注
上一篇
华为再出时尚耳机,FreeBuds Lipstick 2侧翻盖设计引关注
Ruby 与 Go / sha256 hmac base64 编码字符串不匹配
下一篇
Ruby 与 Go / sha256 hmac base64 编码字符串不匹配
查看更多
最新文章
查看更多
课程推荐
  • 前端进阶之JavaScript设计模式
    前端进阶之JavaScript设计模式
    设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
    542次学习
  • GO语言核心编程课程
    GO语言核心编程课程
    本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
    508次学习
  • 简单聊聊mysql8与网络通信
    简单聊聊mysql8与网络通信
    如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
    497次学习
  • JavaScript正则表达式基础与实战
    JavaScript正则表达式基础与实战
    在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
    487次学习
  • 从零制作响应式网站—Grid布局
    从零制作响应式网站—Grid布局
    本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
    484次学习
查看更多
AI推荐
  • AI Make Song:零门槛AI音乐创作平台,助你轻松制作个性化音乐
    AI Make Song
    AI Make Song是一款革命性的AI音乐生成平台,提供文本和歌词转音乐的双模式输入,支持多语言及商业友好版权体系。无论你是音乐爱好者、内容创作者还是广告从业者,都能在这里实现“用文字创造音乐”的梦想。平台已生成超百万首原创音乐,覆盖全球20个国家,用户满意度高达95%。
    14次使用
  • SongGenerator.io:零门槛AI音乐生成器,快速创作高质量音乐
    SongGenerator
    探索SongGenerator.io,零门槛、全免费的AI音乐生成器。无需注册,通过简单文本输入即可生成多风格音乐,适用于内容创作者、音乐爱好者和教育工作者。日均生成量超10万次,全球50国家用户信赖。
    12次使用
  •  BeArt AI换脸:免费在线工具,轻松实现照片、视频、GIF换脸
    BeArt AI换脸
    探索BeArt AI换脸工具,免费在线使用,无需下载软件,即可对照片、视频和GIF进行高质量换脸。体验快速、流畅、无水印的换脸效果,适用于娱乐创作、影视制作、广告营销等多种场景。
    11次使用
  • SEO标题协启动:AI驱动的智能对话与内容生成平台 - 提升创作效率
    协启动
    SEO摘要协启动(XieQiDong Chatbot)是由深圳协启动传媒有限公司运营的AI智能服务平台,提供多模型支持的对话服务、文档处理和图像生成工具,旨在提升用户内容创作与信息处理效率。平台支持订阅制付费,适合个人及企业用户,满足日常聊天、文案生成、学习辅助等需求。
    16次使用
  • Brev AI:零注册门槛的全功能免费AI音乐创作平台
    Brev AI
    探索Brev AI,一个无需注册即可免费使用的AI音乐创作平台,提供多功能工具如音乐生成、去人声、歌词创作等,适用于内容创作、商业配乐和个人创作,满足您的音乐需求。
    17次使用
微信登录更方便
  • 密码登录
  • 注册账号
登录即同意 用户协议隐私政策
返回登录
  • 重置密码