猿问

如何在类型为map[int]的接口上实现接口

我正在尝试构建一个可以在 map[int]T 上实现的函数。我是个新手,想知道这是否可以通过实现接口来完成。


    invoices   map[int]domain.Invoice

    bookings   map[int]domain.Booking

    projects   map[int]domain.Project

所有这些都有以下共同点:


type Invoice struct {

    ID         int

}

type Booking struct {

    ID         int

}

type Project struct {

    ID         int

}

我必须如何继续实现一个函数,通过增加相应类型映射中最后一个项目的 ID 来返回所有发票、预订或项目的下一个 ID?


例如:


func (i *map[int]T) nextID() int {

    return T.ID + 1


qq_花开花谢_0
浏览 137回答 3
3回答

慕姐8265434

收集到所有信息后,我得出以下答案:https://play.golang.org/p/3RAvclRacbh(为了方便命名,我使用了“List”)package mainimport (    "fmt")type Invoice struct{    Name string    Id int}// Interface to implementtype IDer interface {    ID() int}// Interface implementation for Invoicefunc (i Invoice) ID() int { return i.Id }type List struct {    Items    map[int]IDer    last IDer}func (i *List) Add(index int, ider IDer) {    i.Items[index] = ider    i.last = ider}func (i *List) nextID() int {        if i.last == nil {          return 1        }    return i.last.ID() + 1}type Repository struct {    invoices List}func main() {    r := Repository{}    r.invoices = List{         Items: make(map[int]IDer),     }    i := Invoice{}    i.Name = "Test"    i.Id = 1    r.invoices.Add(1, i)    ia := r.invoices.Items[1].(Invoice)     fmt.Println(ia.Name)    fmt.Printf("Next ID: %d\n", r.invoices.nextID())    i2 := Invoice{}    i2.Name = "Test2"    i2.Id = r.invoices.nextID()    r.invoices.Add(i2.Id, i2)    ia2 := r.invoices.Items[i2.Id].(Invoice)        fmt.Println(ia2.Name)    fmt.Printf("Next ID: %d\n", r.invoices.nextID())    i3 := Invoice{}    i3.Name = "Test3"    i3.Id = r.invoices.nextID()    r.invoices.Add(i3.Id, i3)    ia3 := r.invoices.Items[i3.Id].(Invoice)        fmt.Println(ia3.Name)    fmt.Printf("Next ID: %d\n", r.invoices.nextID())}TestNext ID: 2Test2Next ID: 3Test3Next ID: 4Program exited.

蓝山帝景

wondered, if this can be done by implementing an interface当然可以。我不会假装提供正确的惯用方式。但从你的说法来看,所有这些类型都可以归纳为一个IDer接口。为此,他们必须实现一个定义为 的接口type IDer interface { ID() int }。ID()必须为每个结构类型实现此方法。使用通用的兼容实现,您可以定义 a并将map[int]IDer其放入其中Invoice,只要它兼容即可。BookingProject为了防止代码重复,您可以定义一个type IDed struct { ID int}; func (i ID) int {return i.ID},并将其嵌入到您的类型中,例如type Invoice struct { IDed }等等。这样做您仍然可以打电话var x Invoice;  x.ID=1; someid := x.ID;最后,您可以将映射定义为类型type mapOfID map[int]IDer并向其附加方法func (m mapOfID) DoSomethigWithIDers(){...}

炎炎设计

地图本身在跟踪“最后”项目方面做得很差,因为它不是其存储或语义的一部分。解决这个问题的一种方法是跟踪最后一个索引或domain添加到映射中的最后一个对象,以便计算下一个 id:type IDer interface {    ID() int}type IDS struct {    m    map[int]IDer    last IDer}func (i *IDS) Add(index int, ider IDer) {    i.m[index] = ider    i.last = ider}func (i *IDS) nextID() int {    if i.last == nil {      return 1    }    return i.last.ID() + 1}上面的示例结合了 @mh-cbon 的 IDer 接口和跟踪最后添加的 Ider 的能力。domain然后,只要任何对象实现了该接口,就可以将其与它们一起使用IDer:type Invoice struct{}func (i Invoice) ID() int { return 1 }func main() {    ids := &IDS{       m: make(map[int]IDer),    }    ids.Add(1, Invoice{})    fmt.Printf("Next ID: %d\n", ids.nextID())}Next ID: 2Program exited.
随时随地看视频慕课网APP

相关分类

Go
我要回答