登录
首页 >  Golang >  Go问答

如何最简单地记录一个结构中的单个字段?

来源:stackoverflow

时间:2024-02-14 12:33:26 390浏览 收藏

本篇文章给大家分享《如何最简单地记录一个结构中的单个字段?》,覆盖了Golang的常见基础知识,其实一个语言的全部知识点一篇文章是不可能说完的,但希望通过这些问题,让读者对自己的掌握程度有一定的认识(B 数),从而弥补自己的不足,更好的掌握它。

问题内容

假设我有一个函数,它接受 person 的一部分:

type Person struct {
    ID              uuid.UUID   
    FirstName       string
    LastName        string
    ... plus 20 more fields
}

记录时我可能只想记录 id。有没有一种简单的方法可以在不创建另一种类型的情况下做到这一点?我正在使用 logrus。如果我使用 js,我只会使用 map 函数。示例记录器行:

logger.debugf("personprocessor 正在为 people: %v 启动", people)

但这会导致我的日志中产生大量不必要的输出。该 id 足以找到我们正在处理的人。


正确答案


使用reflect包编写一个“通用”函数来提取字段:

func extractfield(s interface{}, name string) []interface{} {
    var result []interface{}
    v := reflect.valueof(s)
    for i := 0; i < v.len(); i++ {
        result = append(result, reflect.indirect(v.index(i)).fieldbyname(name).interface())
    }
    return result
}

像这样使用它:

logger.debugf("personprocessor starting for people: %v", extractfield(persons, "id"))

Run an example on the playground

该功能可以扩展到包括格式化花哨的功能:

// sprintfields prints the field name in slice of struct s 
// using the specified format.
func sprintfields(s interface{}, name string, sep string, format string) string {
    var fields []string
    v := reflect.valueof(s)
    for i := 0; i < v.len(); i++ {
        fields = append(fields, 
            fmt.sprintf(format,
                reflect.indirect(v.index(i)).fieldbyname(name).interface()))
    }
    return strings.join(fields, sep)
}

Try it on the playground

我不确定 logrus,但这适用于标准库:

package main

import (
   "fmt"
   "strconv"
)

type Person struct {
   ID int
   FirstName, LastName string
}

func (p Person) String() string {
   return strconv.Itoa(p.ID)
}

func main() {
   p := Person{9, "First", "Last"}
   fmt.Println(p) // 9
}

以上就是《如何最简单地记录一个结构中的单个字段?》的详细内容,更多关于的资料请关注golang学习网公众号!

声明:本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>