登录
首页 >  Golang >  Go问答

编写一个函数以返回自定义接口切片

来源:stackoverflow

时间:2024-03-02 08:09:24 124浏览 收藏

在IT行业这个发展更新速度很快的行业,只有不停止的学习,才不会被行业所淘汰。如果你是Golang学习者,那么本文《编写一个函数以返回自定义接口切片》就很适合你!本篇内容主要包括##content_title##,希望对大家的知识积累有所帮助,助力实战开发!

问题内容

首先,我想向您介绍一下,因为我觉得我缺少一些 golang 核心概念。

在我的应用程序中,许多模型都有一个名为 getbyuserid 的方法。我创建了需要此方法的接口(usercreatedentity),这样我就可以为每种类型的记录创建控制器的 getuserrecords 方法工厂:

router.handle("/ideas/mine",
        middlewares.authuser(controllers.getminefactory(&models.idea{}))).methods("post")

router.handle("/votes/mine",
        middlewares.authuser(controllers.getminefactory(&models.vote{}))).methods("post")

router.handle("/somenewtype/mine",
        middlewares.authuser(controllers.getminefactory(&models.somenewtype{}))).methods("post")

这就是我的界面的样子:

type usercreatedentity interface {
    getbyuserid(userid uint) []usercreatedentity
}

和实施:

func (idea *idea) getbyuserid(userid uint) []usercreatedentity {
    ideas := []idea{}
    getdb().
        table("ideas").
        /** query removed to make code less confusing **/
        scan(ideas)

    return ideas
}

显然,它不起作用(带有指针切片的版本也不起作用)。问题是 - 如果我只返回一条记录,这段代码就可以工作 - 就像那样(显然也改变了界面中的签名):

func (idea *idea) getbyuserid(userid uint) usercreatedentity {
    idea := &idea{}
    getdb().
        table("ideas").
        /** query removed to make code less confusing **/
        first(idea)

    return idea
}

如何让它像切片一样工作?正如我所说,我怀疑我错过了一些重要的知识。如此深入的解释会很棒。

解决方案:

func (idea *Idea) GetByUserId(userId uint) []UserCreatedEntity {
    ideas := []*Idea{}
    GetDB().
        Table("ideas").
        Select("problems.name AS problem_name, ideas.id, ideas.problem_id, ideas.action_description, ideas.results_description, ideas.money_price, ideas.time_price, ideas.is_published").
        Joins("INNER JOIN problems ON ideas.problem_id = problems.id").
        Where("ideas.user_id = ?", userId).
        Scan(&ideas)

    uces := make([]UserCreatedEntity, len(ideas))
    for i, idea := range ideas {
        uces[i] = idea
    }
    return uces
}

解决方案


界面是动态的。涉及接口的复合类型则不然。

usercreatedentity 是一个接口,并且 idea 满足该接口,因此您可以从签名返回类型为 usercreatedentity 的函数返回 idea

[]usercreatedentityusercreatedentity 的一部分,而不是接口。唯一可以返回的类型是 []usercreatedentity[]idea 是一种不同的类型(idea 的切片)。您可以使用 idea 元素填充 []usercreatedentity,因为每个元素的类型都是 usercreatedentity,这又是一个接口,并且允许使用 idea

类似地,func() usercreatedentity 是一个类型“返回 usercreatedentity 的函数”。您不能替换 func() idea,因为这是不同的类型。但是您可以从 func() usercreatedentity 返回 idea,因为 ideausercreatedentity

如果您在这里没有使用 scan(可能使用反射),解决方法是将本地切片声明为 []usercreatedentity 而不是 []idea。由于您使用的是 scan,因此您必须扫描到 []idea,然后迭代它以将所有元素复制到 []usercreatedentity 并返回。

在编程语言理论中,这称为方差it is not supported in Go。有关更多详细信息,请参阅 this proposal

具体来说,返回类型不是协变的。即使 t 实现了 it 的切片也不会实现 i 的切片。

上面链接的常见问题解答条目提出了此解决方法:

t := []int{1, 2, 3, 4}
s := make([]interface{}, len(t))
for i, v := range t {
    s[i] = v
}

尽管根据您的情况,正确的解决方案可能有所不同。

今天关于《编写一个函数以返回自定义接口切片》的内容就介绍到这里了,是不是学起来一目了然!想要了解更多关于的内容请关注golang学习网公众号!

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