设为首页收藏本站
网站公告 | 这是第一条公告
     

 找回密码
 立即注册
缓存时间06 现在时间06 缓存数据 做一个积极向上的人,读温柔的句子,见阳光的人,眼里全是温柔和笑意。

做一个积极向上的人,读温柔的句子,见阳光的人,眼里全是温柔和笑意。

查看: 1034|回复: 2

Go语言断言和类型查询的实现

[复制链接]

  离线 

TA的专栏

  • 打卡等级:初来乍到
  • 打卡总天数:4
  • 打卡月天数:0
  • 打卡总奖励:59
  • 最近打卡:2023-08-16 05:45:13
等级头衔

等級:晓枫资讯-列兵

在线时间
0 小时

积分成就
威望
0
贡献
34
主题
28
精华
0
金钱
160
积分
64
注册时间
2023-8-12
最后登录
2025-3-15

发表于 2024-2-25 16:42:24 | 显示全部楼层 |阅读模式
目录


  • 1、类型断言
  • 2、类型查询

1、类型断言

类型断言(Type Assertion)是一个使用在接口值上的操作,用于检查接口类型变量所持有的值是否实现了期望的接口或者具体的类型。
在Go语言中类型断言的语法格式如下:
  1. // i.(TypeNname)
  2. value, ok := x.(T)
复制代码
其中,x 表示一个接口的类型,如果是具体类型变量,则编译器会报
  1. non-interface type xxx on left
复制代码
,T 表示一个具体的类型(也可为接口类型)。
该断言表达式会返回 x 的值(也就是 value)和一个布尔值(也就是 ok),可根据该布尔值判断 x 是否为 T 类型:

  • 如果 T 是具体某个类型,类型断言会检查 x 的动态类型是否等于具体类型 T。如果检查成功,类型断言返回的结果是 x 的动态值,其类型是 T。
  • 如果 T 是接口类型,类型断言会检查 x 的动态类型是否满足 T。如果检查成功,x 的动态值不会被提取,返回值是一个类型为 T 的接口值。
  • 无论 T 是什么类型,如果 x 是 nil 接口值,类型断言都会失败。
示例代码如下:
  1. package main

  2. import (
  3.         "fmt"
  4. )

  5. func main() {
  6.         var x interface{}
  7.         x = 10
  8.         value, ok := x.(int)
  9.         // 10,true
  10.         fmt.Print(value, ",", ok)
  11. }
复制代码
运行结果如下:
  1. # 程序结果10,true
复制代码
需要注意如果不接收第二个参数也就是上面代码中的 ok,断言失败时会直接造成一个 panic,如果 x 为 nil 同样也会 panic。
示例代码如下:
  1. package main

  2. import (
  3.         "fmt"
  4. )

  5. func main() {
  6.         var x interface{}
  7.         x = "Hello"
  8.         value := x.(int)
  9.         fmt.Println(value)
  10. }
复制代码
运行结果如下:
  1. # 输出结果panic: interface conversion: interface {} is string, not int
复制代码
接口断言通常可以使用 comma,ok 语句来确定接口是否绑定某个实例类型,或者判断接口绑定的实例类型是否实现另一个接口。
  1. re,ok := body.(io.ReadCloser)
  2. if,ok :=  r.Body.(*maxBytesReader);
复制代码
2、类型查询

接口类型查询的语法格式如下:
  1. switch v := i.(type){
  2.         case typel:
  3.                 XXXX
  4.         case type2:
  5.                 XXXX
  6.         default:
  7.                 XXXX
  8. }
复制代码
接口查询有两层语义,一是查询一个接口变量底层绑定的底层变量的具体类型是什么,二是查询接口变量绑定的底
层变量是否还实现了其他接口。
(1)、i 必须是接口类型
具体类型实例的类型是静态的,在类型声明后就不再变化,所以具体类型的变量不存在类型查询,类型查询一定是
对一个接口变量进行操作。也就是说,上文中的i必须是接口变量,如果i是未初始化接口变量,则v的值是 nil。
  1. package main

  2. import (
  3.         "fmt"
  4.         "io"
  5. )

  6. func main() {
  7.         var i io.Reader
  8.         //此处i是为未初始化的接口变量,所以v为nil
  9.         switch v := i.(type) {
  10.         case nil:
  11.                 //<nil>
  12.                 fmt.Printf("%T\n", v)
  13.         default:
  14.                 fmt.Printf("default")
  15.         }
  16. }
复制代码
(2)、case 字句后面可以跟非接口类型名,也可以跟接口类型名,匹配是按照 case 子句的顺序进行的。

  • 如果 case 后面是一个接口类型名,且接口变量i绑定的实例类型实现了该接口类型的方法,则匹配成,v的类型是接口类型,v底层绑定的实例是i绑定具体类型实例的副本。
  • 如果 case 后面是一个具体类型名,且接口变量i绑定的实例类型和该具体类型相同,则匹配成功,此时v 就是该具体类型变量,v的值是i绑定的实例值的副本。
  • 如果 case 后面跟着多个类型,使用逗号分隔,接口变量i绑定的实例类型只要和其中一个类型匹配,则直接使用o赋值给v,相当于 v:=o。这个语法有点奇怪,按理说编译器不应该允许这种操作,语言实现者可能想让type switch 语句和普通的 switch 语句保持一样的语法规则,允许发生这种情况。
  • 如果所有的case字句都不满足,则执行 default 语句,此时执行的仍然是 v:=o,最终v的值是o。此时使用v没有任何意义。
  • fallthrough 语句不能在 Type Switch 语句中使用。
  1. package main

  2. import (
  3.         "fmt"
  4.         "io"
  5.         "log"
  6.         "os"
  7. )

  8. func main() {
  9.         var i io.Reader
  10.         // 此处i是为未初始化的接口变量,所以v为nil
  11.         switch v := i.(type) {
  12.         case nil:
  13.                 // <nil>
  14.                 fmt.Printf("%T\n", v)
  15.         default:
  16.                 fmt.Printf("default")
  17.         }
  18.         f, err := os.OpenFile("notes.txt", os.O_RDWR|os.O_CREATE, 0755)
  19.         if err != nil {
  20.                 log.Fatal(err)
  21.         }
  22.         defer f.Close()
  23.         i = f
  24.         switch v := i.(type) {
  25.         // i的绑定的实例是*osFile类型,实现了io.ReadWriter接口,所以下面case匹配成功
  26.         case io.ReadWriter:
  27.                 // v是io.ReadWriter接口类型,所以可以调用Write方法
  28.                 v.Write([]byte("io.ReadWriter\n"))
  29.                 // Type Switch Result: io.ReadWriter
  30.                 fmt.Println("Type Switch Result: io.ReadWriter")
  31.         // 由于上一个case已经匹配,就算这个case也匹配,也不会走到这里
  32.         case *os.File:
  33.                 v.Write([]byte("*os.File\n"))
  34.                 fmt.Println("Type Switch Result: *os.File")
  35.                 //这里可以调用具体类型方法
  36.                 v.Sync()
  37.         default:
  38.                 fmt.Println("Type Switch Result: unknown")
  39.                 return
  40.         }

  41.         switch v := i.(type) {
  42.         // 匹配成功,v的类型就是具体类型*os.File
  43.         case *os.File:
  44.                 v.Write([]byte("*os.File\n"))
  45.                 // Type Switch Result: *os.File
  46.                 fmt.Println("Type Switch Result: *os.File")
  47.                 v.Sync()
  48.         //由于上一个case已经匹配,就算这个case也匹配,也不会走到这里
  49.         case io.ReadWriter:
  50.                 //v是io.ReadWriter接口类型,所以可以调用Write方法
  51.                 v.Write([]byte("io.ReadWriter\n"))
  52.                 fmt.Println("Type Switch Result: io.ReadWriter")
  53.         default:
  54.                 fmt.Println("Type Switch Result: unknown")
  55.                 return
  56.         }

  57.         switch v := i.(type) {
  58.         //多个类型,f满足其中任何一个就算匹配
  59.         case *os.File, io.ReadWriter:
  60.                 // 此时相当于执行力v := i ,v和i是等价的,使用v没有意义。
  61.                 if v == i {
  62.                         // true
  63.                         fmt.Println(true)
  64.                 }
  65.         default:
  66.                 return
  67.         }
  68. }
复制代码
  1. # 程序输出<nil>Type Switch Result: io.ReadWriterType Switch Result: *os.Filetrue 
复制代码
  1. package main

  2. import (
  3.         "fmt"
  4. )

  5. func main() {
  6.         var a int
  7.         a = 10
  8.         // the type of a is int
  9.         getType(a)
  10. }

  11. func getType(a interface{}) {
  12.         switch a.(type) {
  13.         case int:
  14.                 fmt.Println("the type of a is int")
  15.         case string:
  16.                 fmt.Println("the type of a is string")
  17.         case float64:
  18.                 fmt.Println("the type of a is float")
  19.         default:
  20.                 fmt.Println("unknown type")
  21.         }
  22. }
复制代码
  1. # 程序输出the type of a is int
复制代码
到此这篇关于Go语言断言和类型查询的实现的文章就介绍到这了,更多相关Go语言断言和类型查询内容请搜索晓枫资讯以前的文章或继续浏览下面的相关文章希望大家以后多多支持晓枫资讯!

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!
晓枫资讯-科技资讯社区-免责声明
免责声明:以上内容为本网站转自其它媒体,相关信息仅为传递更多信息之目的,不代表本网观点,亦不代表本网站赞同其观点或证实其内容的真实性。
      1、注册用户在本社区发表、转载的任何作品仅代表其个人观点,不代表本社区认同其观点。
      2、管理员及版主有权在不事先通知或不经作者准许的情况下删除其在本社区所发表的文章。
      3、本社区的文章部分内容可能来源于网络,仅供大家学习与参考,如有侵权,举报反馈:点击这里给我发消息进行删除处理。
      4、本社区一切资源不代表本站立场,并不代表本站赞同其观点和对其真实性负责。
      5、以上声明内容的最终解释权归《晓枫资讯-科技资讯社区》所有。
http://bbs.yzwlo.com 晓枫资讯--游戏IT新闻资讯~~~

  离线 

TA的专栏

等级头衔

等級:晓枫资讯-列兵

在线时间
0 小时

积分成就
威望
0
贡献
0
主题
0
精华
0
金钱
13
积分
6
注册时间
2022-12-25
最后登录
2022-12-25

发表于 2024-10-25 12:01:35 | 显示全部楼层
感谢楼主分享。
http://bbs.yzwlo.com 晓枫资讯--游戏IT新闻资讯~~~

  离线 

TA的专栏

等级头衔

等級:晓枫资讯-列兵

在线时间
0 小时

积分成就
威望
0
贡献
0
主题
0
精华
0
金钱
21
积分
22
注册时间
2022-12-25
最后登录
2022-12-25

发表于 2025-4-1 13:59:18 | 显示全部楼层
顶顶更健康!!!
http://bbs.yzwlo.com 晓枫资讯--游戏IT新闻资讯~~~
严禁发布广告,淫秽、色情、赌博、暴力、凶杀、恐怖、间谍及其他违反国家法律法规的内容。!晓枫资讯-社区
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

1楼
2楼
3楼

手机版|晓枫资讯--科技资讯社区 本站已运行

CopyRight © 2022-2025 晓枫资讯--科技资讯社区 ( BBS.yzwlo.com ) . All Rights Reserved .

晓枫资讯--科技资讯社区

本站内容由用户自主分享和转载自互联网,转载目的在于传递更多信息,并不代表本网赞同其观点和对其真实性负责。

如有侵权、违反国家法律政策行为,请联系我们,我们会第一时间及时清除和处理! 举报反馈邮箱:点击这里给我发消息

Powered by Discuz! X3.5

快速回复 返回顶部 返回列表