为了账号安全,请及时绑定邮箱和手机立即绑定

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

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

Go
qq_花开花谢_0 2023-07-10 14:35:41
我正在尝试构建一个可以在 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
查看完整描述

3 回答

?
慕姐8265434

TA贡献1813条经验 获得超2个赞

收集到所有信息后,我得出以下答案:

https://play.golang.org/p/3RAvclRacbh

(为了方便命名,我使用了“List”)

package main


import (

    "fmt"

)


type Invoice struct{

    Name string

    Id int

}


// Interface to implement

type IDer interface {

    ID() int

}

// Interface implementation for Invoice

func (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())

}


Test

Next ID: 2

Test2

Next ID: 3

Test3

Next ID: 4


Program exited.


查看完整回答
反对 回复 2023-07-10
?
蓝山帝景

TA贡献1843条经验 获得超7个赞

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(){...}


查看完整回答
反对 回复 2023-07-10
?
炎炎设计

TA贡献1808条经验 获得超4个赞

地图本身在跟踪“最后”项目方面做得很差,因为它不是其存储或语义的一部分。解决这个问题的一种方法是跟踪最后一个索引或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: 2


Program exited.


查看完整回答
反对 回复 2023-07-10
  • 3 回答
  • 0 关注
  • 143 浏览
慕课专栏
更多

添加回答

举报

0/150
提交
取消
意见反馈 帮助中心 APP下载
官方微信