一文搞懂Golang中iota的用法和原理

 更新时间:2022年08月30日 08:18:12   作者:yi个俗人  
我们知道iota是go语言的常量计数器,本文尝试全面总结其使用用法以及其实现原理,需要的朋友可以参考以下内容,希望对大家有所帮助

前言

我们知道iota是go语言的常量计数器,只能在常量的const表达式中使用,在const关键字出现的时将被重置为0const中每新增一行常量声明iota值自增1(iota可以理解为const语句块中的行索引),使用iota可以简化常量的定义,但其规则必须要牢牢掌握,否则在我们开发中可能会造成误解,本文尝试全面总结其使用用法以及其实现原理,需要的朋友可以参考以下内容,希望对大家有帮助。

iota的使用

iota在const关键字出现时将被重置为0

iota只能在常量的表达式中使用,iotaconst关键字出现时将被重置为0。不同const定义块互不干扰。

//const关键字出现时将被重置为0
const (
    a = iota  //0
    b         //1
)
//不同const定义块互不干扰
const (
    c = iota //0
)

按行计数

const每新增一行常量声明,iota计数一次,可以当做const语句中的索引,常用于定义枚举数据。

const (
    n1 = iota //0
    n2 		  //1
    n3 		  //2
    n4 		  //3
)

所有注释行和空行全部忽略

所有注释行和空行在编译时期首先会被清除,所以空行不计数。

const (
    a = iota  //0
    
    b		  //1
    
    //此行是注释
    c         //2
)

跳值占位

如果某个值不需要,可以使用占位 “_”,它不是空行,会进行计数,起到跳值作用。

const (
    a = iota //0
    _
    b        //2
)

多个iota

同一const块出现多个iota,只会按照行数计数,不会重新计数。

const (
    a = iota     // a=0
    b = iota     // b=1
    c = iota     // c=2
  )

一行多个iota

一行多个iota,分别计数。

const (
    a, b = iota, iota // a=0,b=0
    c, d              // c=1,d=1
)

首行插队

开头插队会进行计数。

const (
    a = 100 //  a=100
    b = iota // b=1
    c = iota // c=2
    d        // d=3
)

中间插队

中间插队会进行计数。

const (
    a = iota // a=0
    b = 100  // b=100
    c = iota // c=2
    d        // d=3
)

没有表达式的常量定义复用上一行的表达式

const (
   a = iota       //  iota = 0 
   b = 1 + iota  // iota = 1
   c           // iota = 2
)

实现原理

iota定义

iota 源码在 Go 语言代码库中的定义位于内建文件 go/src/builtin/builtin.go 中:

const iota = 0 // Untyped int.iota

在这里声明了一个常量标识符,它的值是0;iota只是一个简单的整数0,为什么能作为常量计数器进行自增的,我们再看一下const的实现。

const

const 块中每一行在 Go 中使用 spec 数据结构描述, spec 声明如下:

ValueSpec struct {
 Doc *CommentGroup // associated documentation; or nil
 Names []*Ident // value names (len(Names) > 0)
 Type Expr // value type; or nil
 Values []Expr // initial values; or nil
 Comment *CommentGroup // line comments; or nil
}

在这个结构体中有一个切片 ValueSpec.Names,此切片中保存了一行中定义的常量,如果一行定义N个常量,那么 ValueSpec.Names 切片长度即为N。

const块实际上是spec类型的切片,用于表示const中的多行。

编译期间构造常量时的伪算法如下:

for iota, spec := range ValueSpecs {
 for i, name := range spec.Names {
 obj := NewConst(name, iota...) //此处将iota传入,用于构造常量
 ...
 }
}

iota实际上是遍历const块的索引,每行中即便多次使用iota,其值也不会递增。

到此这篇关于一文搞懂Golang中iota的用法和原理的文章就介绍到这了,更多相关Golang iota内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • GoLang bytes.Buffer基础使用方法详解

    GoLang bytes.Buffer基础使用方法详解

    Go标准库中的bytes.Buffer(下文用Buffer表示)类似于一个FIFO的队列,它是一个流式字节缓冲区,我们可以持续向Buffer尾部写入数据,从Buffer头部读取数据。当Buffer内部空间不足以满足写入数据的大小时,会自动扩容
    2023-03-03
  • Go语言函数学习教程

    Go语言函数学习教程

    这篇文章主要介绍了Go语言函数基本用法,结合实例形式分析了Go语言函数的格式、定义、使用方法与相关注意事项,需要的朋友可以参考下
    2016-07-07
  • 一文带你了解Golang中interface的设计与实现

    一文带你了解Golang中interface的设计与实现

    本文就来详细说说为什么说 接口本质是一种自定义类型,以及这种自定义类型是如何构建起 go 的 interface 系统的,感兴趣的小伙伴可以跟随小编一起学习一下
    2023-01-01
  • golang针对map的判断,删除操作示例

    golang针对map的判断,删除操作示例

    这篇文章主要介绍了golang针对map的判断,删除操作,结合具体实例形式分析了Go语言map判断与删除相关操作技巧,需要的朋友可以参考下
    2017-03-03
  • golang 字符串拼接方法对比分析

    golang 字符串拼接方法对比分析

    这篇文章主要为大家介绍了golang 字符串拼接方法对比分析,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-09-09
  • golang 实现Location跳转方式

    golang 实现Location跳转方式

    这篇文章主要介绍了golang 实现Location跳转方式,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2021-05-05
  • Go语言建议多使用切片少使用数组原理探究

    Go语言建议多使用切片少使用数组原理探究

    这篇文章主要为大家介绍了Go语言建议多使用切片少使用数组原理探究,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2024-01-01
  • Golang加权轮询负载均衡的实现

    Golang加权轮询负载均衡的实现

    负载均衡器在向后端服务分发流量负载时可以使用几种策略。本文主要介绍了Golang加权轮询负载均衡,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-06-06
  • logrus日志自定义格式操作

    logrus日志自定义格式操作

    这篇文章主要介绍了logrus日志自定义格式操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-11-11
  • Go 互斥锁和读写互斥锁的实现

    Go 互斥锁和读写互斥锁的实现

    本文主要介绍了Go 互斥锁和读写互斥锁的实现,文中通过示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-11-11

最新评论