如何使用泛型将接口转换为指定类型

huangapple go评论89阅读模式
英文:

How to transfer interface to specified type by using generic

问题

这是一个声明接口和实现接口的多个结构体的代码示例。

  1. type DataInterface interface {
  2. Get(string) string
  3. }
  4. type DataA struct {
  5. d map[string]string
  6. }
  7. func (d *DataA) Get(key string) string {
  8. return d.d[key]
  9. }
  10. func (d *DataA) GetId() string {
  11. return d.Get("id")
  12. }
  13. type DataB struct {
  14. d map[string]string
  15. }
  16. func (d *DataB) Get(key string) string {
  17. return d.d[key]
  18. }
  19. func (d *DataB) GetFile() string {
  20. return d.Get("file")
  21. }
  22. type DataC...
  23. 还包括`DataC,D,E...`
  24. 我将把这些`DataX`结构体的实例存储在`type DataSlice []DataInterface`
  25. 现在如果我想获取`DataX`可以这样做
  26. ```go
  27. type DataSlice []DataInterface
  28. func (d DataSlice) GetA() []*DataA {
  29. var ret []*DataA
  30. for _, di := range d {
  31. if v, ok := di.(*DataA); ok {
  32. ret = append(ret, v)
  33. }
  34. }
  35. return ret
  36. }
  37. func (d DataSlice) GetB() []*DataB {
  38. var ret []*DataB
  39. for _, di := range d {
  40. if v, ok := di.(*DataB); ok {
  41. ret = append(ret, v)
  42. }
  43. }
  44. return ret
  45. }
  46. func (d DataSlice) GetC() .....

显然,这里有很多重复的代码:

  1. var ret []*DataX
  2. for _, di := range d {
  3. if v, ok := di.(*DataX); ok {
  4. ret = append(ret, v)
  5. }
  6. }

因此,我考虑使用泛型来解决这个问题,然后定义了这个函数:

  1. func GetDataX[T any] (d DataInterface) *T {
  2. return d.(*T)
  3. }

但是出现了错误:Impossible type assertion: '*T' does not implement 'DataInterface'

所以,我想知道这种方式真的不可能吗?还是可以通过其他方式完成?

英文:

There is an Interface declare and many structs that implement it

  1. type DataInterface interface {
  2. Get(string) string
  3. }
  4. type DataA struct {
  5. d map[string]string
  6. }
  7. func (d *DataA) Get(key string) string {
  8. return d.d[key]
  9. }
  10. func (d *DataA) GetId() string {
  11. return d.Get("id")
  12. }
  13. type DataB struct {
  14. d map[string]string
  15. }
  16. func (d *DataB) Get(key string) string {
  17. return d.d[key]
  18. }
  19. func (d *DataB) GetFile() string {
  20. return d.Get("file")
  21. }
  22. type DataC...

Also includes DataC,D,E...

and I will store these DataX structs instance into a type DataSlice []DataInterface

Now, If I want to get DataX , I can do this:

  1. type DataSlice []DataInterface
  2. func (d DataSlice) GetA() []*DataA {
  3. var ret []*DataA
  4. for _, di := range d {
  5. if v, ok := di.(*DataA); ok {
  6. ret = append(ret, v)
  7. }
  8. }
  9. return ret
  10. }
  11. func (d DataSlice) GetB() []*DataB {
  12. var ret []*DataB
  13. for _, di := range d {
  14. if v, ok := di.(*DataB); ok {
  15. ret = append(ret, v)
  16. }
  17. }
  18. return ret
  19. }
  20. func (d DataSlice) GetC() .....

Obviously there's a lot of repetitive code here:

  1. var ret []*DataX
  2. for _, di := range d {
  3. if v, ok := di.(*DataX); ok {
  4. ret = append(ret, v)
  5. }
  6. }

So I think about that I can use generic to slove this, then I define this function:

  1. func GetDataX[T any] (d DataInterface) *T {
  2. return d.(*T)
  3. }

but got error: Impossible type assertion: '*T' does not implement 'DataInterface

So, I want to know is this way really impossible? Or it could be completed by the other way?

答案1

得分: 1

你应该能够使用以下代码来满足你的需求:

  1. package main
  2. import "fmt"
  3. // 接口
  4. type DataInterface interface {
  5. Get(string) string
  6. }
  7. // 实现接口的结构体
  8. type DataA struct {
  9. d map[string]string
  10. }
  11. func (d DataA) Get(key string) string {
  12. return d.d[key]
  13. }
  14. type DataB struct {
  15. d map[string]string
  16. }
  17. func (d DataB) Get(key string) string {
  18. return d.d[key]
  19. }
  20. type DataSlice []DataInterface
  21. func GetDataX[T any](d DataInterface) T {
  22. return d.(T)
  23. }
  24. func main() {
  25. a := DataA{map[string]string{"a": "1"}}
  26. b := DataB{map[string]string{"b": "2"}}
  27. ds := DataSlice{a, b}
  28. for _, v := range ds {
  29. if value, ok := v.(DataA); ok {
  30. fmt.Printf("A\t%q\n", GetDataX[DataA](value))
  31. continue
  32. }
  33. if value, ok := v.(DataB); ok {
  34. fmt.Printf("B\t%q\n", GetDataX[DataB](value))
  35. continue
  36. }
  37. // 在这里添加未知类型的处理逻辑
  38. }
  39. }

首先,我简化了代码,只考虑了DataADataB结构体。然后,我将指针接收器更改为值接收器,因为你不会更改传递给方法的实际实例的状态。由于这个改变,GetDataX成功工作,你可以获取所有类似结构体的信息。

如果这解决了你的问题,或者你需要其他帮助,请告诉我,谢谢!

英文:

You should be able to handle your needs with the following code:

  1. package main
  2. import "fmt"
  3. // interface
  4. type DataInterface interface {
  5. Get(string) string
  6. }
  7. // struct implementing the interface
  8. type DataA struct {
  9. d map[string]string
  10. }
  11. func (d DataA) Get(key string) string {
  12. return d.d[key]
  13. }
  14. type DataB struct {
  15. d map[string]string
  16. }
  17. func (d DataB) Get(key string) string {
  18. return d.d[key]
  19. }
  20. type DataSlice []DataInterface
  21. func GetDataX[T any](d DataInterface) T {
  22. return d.(T)
  23. }
  24. func main() {
  25. a := DataA{map[string]string{"a": "1"}}
  26. b := DataB{map[string]string{"b": "2"}}
  27. ds := DataSlice{a, b}
  28. for _, v := range ds {
  29. if value, ok := v.(DataA); ok {
  30. fmt.Printf("A\t%q\n", GetDataX[DataA](value))
  31. continue
  32. }
  33. if value, ok := v.(DataB); ok {
  34. fmt.Printf("B\t%q\n", GetDataX[DataB](value))
  35. continue
  36. }
  37. // add unknown type handling logic here
  38. }
  39. }

First, I simplified the code to take into consideration only the DataA and DataB structs. Then, I changed the pointer receivers to value receivers as you're not going to change the state of the actual instance passed to the methods. Thanks to this change the GetDataX works successfully and you're able to get the info for all of your similar structs.

Let me know if this solves your issues or if you need something else, thanks!

huangapple
  • 本文由 发表于 2023年2月14日 19:20:13
  • 转载请务必保留本文链接:https://go.coder-hub.com/75447075.html
匿名

发表评论

匿名网友

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定