泛型与嵌套映射

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

Generics with nested maps

问题

目前我正在使用类似这样的代码:

  1. package hello
  2. type object map[string]interface{}
  3. func (o object) get(key string) object {
  4. val, _ := o[key].(object)
  5. return val
  6. }
  7. func (o object) getInt(key string) int {
  8. val, _ := o[key].(int)
  9. return val
  10. }

但问题是,我必须为每种要返回的类型编写一个函数。我尝试使用类似这样的代码:

  1. package hello
  2. type object map[string]interface{}
  3. // 语法错误:方法不能有类型参数
  4. func (o object) get[T object|int](key string) T {
  5. val, _ := o[key].(T)
  6. return val
  7. }

然后我尝试了这样的代码:

  1. package hello
  2. type object map[string]interface{}
  3. type token[T object|int] struct {
  4. in object
  5. out T
  6. }
  7. func (t token[T]) get(key string) token[T] {
  8. t.out, _ = t.in[key].(T)
  9. return t
  10. }

这段代码可以编译通过,但由于in从未更新,我认为我无法进行链式调用,例如对于嵌套的映射:

  1. something.get("one").get("two").get("three")

是否有可能使用泛型来实现与我的原始代码类似的结果,但不需要复制粘贴函数?

英文:

Currently I am using some code like this:

  1. package hello
  2. type object map[string]interface{}
  3. func (o object) get(key string) object {
  4. val, _ := o[key].(object)
  5. return val
  6. }
  7. func (o object) getInt(key string) int {
  8. val, _ := o[key].(int)
  9. return val
  10. }

but the issue is, I have to write a function for every type I want to return. I
tried using something like this:

  1. package hello
  2. type object map[string]interface{}
  3. // syntax error: method must have no type parameters
  4. func (o object) get[T object|int](key string) T {
  5. val, _ := o[key].(T)
  6. return val
  7. }

then I did this:

  1. package hello
  2. type object map[string]interface{}
  3. type token[T object|int] struct {
  4. in object
  5. out T
  6. }
  7. func (t token[T]) get(key string) token[T] {
  8. t.out, _ = t.in[key].(T)
  9. return t
  10. }

which compiles, but since in never gets updated, I don't think I could do chaining, for a nested map:

  1. something.get("one").get("two").get("three")

Is it possible to do something with generics, to give a similar result to my original code, but without having the copy paste functions?

答案1

得分: 1

我想我弄明白了。你可以创建一个包装类型,它包含当前对象以及输出值。如果有其他想法,我也很感兴趣:

  1. package main
  2. type object map[string]interface{}
  3. type token[T any] struct {
  4. object
  5. value T
  6. }
  7. func newToken[T any](m object) token[T] {
  8. return token[T]{object: m}
  9. }
  10. func (t token[T]) get(key string) token[T] {
  11. switch val := t.object[key].(type) {
  12. case object:
  13. t.object = val
  14. case T:
  15. t.value = val
  16. }
  17. return t
  18. }

示例:

  1. package main
  2. func main() {
  3. obj := object{
  4. "one": object{
  5. "two": object{"three": 3},
  6. },
  7. }
  8. three := newToken[int](obj).get("one").get("two").get("three")
  9. println(three.value == 3)
  10. }
英文:

I think I figured it out. You can create a wrapper type, that holds the current
object, as well as the output value. If anyone has other ideas, I am interested
in them as well:

  1. package main
  2. type object map[string]interface{}
  3. type token[T any] struct {
  4. object
  5. value T
  6. }
  7. func newToken[T any](m object) token[T] {
  8. return token[T]{object: m}
  9. }
  10. func (t token[T]) get(key string) token[T] {
  11. switch val := t.object[key].(type) {
  12. case object:
  13. t.object = val
  14. case T:
  15. t.value = val
  16. }
  17. return t
  18. }

Example:

  1. package main
  2. func main() {
  3. obj := object{
  4. "one": object{
  5. "two": object{"three": 3},
  6. },
  7. }
  8. three := newToken[int](obj).get("one").get("two").get("three")
  9. println(three.value == 3)
  10. }

huangapple
  • 本文由 发表于 2022年3月18日 00:50:51
  • 转载请务必保留本文链接:https://go.coder-hub.com/71516163.html
匿名

发表评论

匿名网友

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

确定