登录
首页 >  Golang >  Go问答

非导出类型的类型断言

来源:stackoverflow

时间:2024-03-17 18:27:28 170浏览 收藏

在使用第三方库时,有时会遇到无法直接断言类型的非导出类型。这篇文章介绍了一种使用反射来解决此问题的方法。反射可以将值的类型与目标类型的描述符进行比较,从而过滤掉不需要的值。通过获取目标函数的返回类型的类型描述符,可以避免直接调用函数来获取其类型,从而防止不必要的副作用。

问题内容

我正在使用第 3 方包,它允许您通过导出函数创建某种非导出类型的结构。

package squirrel

type expr struct {
    sql string
    args []interface{}
}

func expr(sql string, args ...interface{}) expr {
    return expr{sql: sql, args: args}
}

由于该库的某些其他函数接受数据的方式,我最终得到了这样的地图:

m := map[string]interface{} {
    "col1": 123,
    "col2": "a_string",
    "col3": expr("now()"),
}

但由于该库中的函数不同,我需要从此地图中过滤掉所有 squirrel.expr

显然,我无法通过这样做直接断言类型:

filtered := make(map[string]interface{})
for k, v := range m {
    switch v.(type) {
    case squirrel.expr:
        continue
    default:
        filtered[k] = v
    }
}

还有其他方法可以达到相同的结果吗?


解决方案


您可以使用反射将值的类型与 squirrel.expr 的类型进行比较。这里的type是指reflect.Type个描述符,由reflect.TypeOf()获取。

例如:

m := map[string]interface{}{
    "col1": 123,
    "col2": "a_string",
    "col3": squirrel.expr("now()"),
}
fmt.println(m)

exprtype := reflect.typeof(squirrel.expr(""))

filtered := make(map[string]interface{})
for k, v := range m {
    if reflect.typeof(v) == exprtype {
        continue
    }
    filtered[k] = v
}
fmt.println(filtered)

这将输出:

map[col1:123 col2:a_string col3:{now() []}]
map[col1:123 col2:a_string]

注意:

通过传递 squirrel.expr() 调用(类型为 squirrel.expr)的返回值,我们获得了要过滤的值的 reflect.type 描述符。在这种情况下这很好,但是如果仅仅为了获取类型而调用这个函数是不可行的(例如,调用有必须避免的副作用),我们可以避免这种情况。我们可以使用反射来获取squirrel.expr函数本身的reflect.type描述符,并获取其返回类型的类型描述符。可以这样做:

exprType := reflect.TypeOf(squirrel.Expr).Out(0)

终于介绍完啦!小伙伴们,这篇关于《非导出类型的类型断言》的介绍应该让你收获多多了吧!欢迎大家收藏或分享给更多需要学习的朋友吧~golang学习网公众号也会发布Golang相关知识,快来关注吧!

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