广告
返回顶部
首页 > 资讯 > 后端开发 > GO >golang内置函数len的小技巧
  • 498
分享到

golang内置函数len的小技巧

2024-04-02 19:04:59 498人浏览 八月长安
摘要

len是很常用的内置函数,可以测量字符串、slice、array、channel以及map的长度/元素个数。 不过你真的了解len吗?也许还有一些你不知道的小知识。 我们来看一道Go

len是很常用的内置函数,可以测量字符串、slice、array、channel以及map的长度/元素个数。

不过你真的了解len吗?也许还有一些你不知道的小知识。

我们来看一道Go101的题目,这题也被GO语言爱好者周刊转载:


package main

import "fmt"

func main() {
    var x *struct {
        s [][32]byte
    }
    
    fmt.Println(len(x.s[99]))
}

题目问你这段代码的运行结果,选项有编译错误、panic、32和0。

我们分析一下,别看x的声明定义一大长串,实际上就是定义了一个有个[][32]byte的结构体,然后x是这个结构体的指针。

然后我们没有初始化x,所以x是一个值为nil的指针。看到这里你也许以及有答案了,对nil指针解引用访问它的成员s,那不就是panic嘛。即使引用x的成员合法,我们的s也没有初始化,访问没有初始化的slice也会panic。

然而这么想你就错了,代码的实际运行结果是32!

为什么呢?我们看看len的帮助文档:

For some arguments, such as a string literal or a simple array expression, the result can be a constant. See the Go language specification's "Length and capacity" section for details.

这句话很重要,对于结果是数组的表达式,len可能会是一个编译期常量,而且数组类型的长度在编译期是可知的,所以熟悉c++的朋友大概会立刻想到这样的常量是不需要进行实际求值的,简单类型推导即可获得。不过口说无凭,我们看看spec里的描述:

The expression len(s) is constant if s is a string constant. The expressions len(s) and cap(s) are constants if the type of s is an array or pointer to an array and the expression s does not contain channel receives or (non-constant) function calls; in this case s is not evaluated. Otherwise, invocations of len and cap are not constant and s is evaluated.

如果表达式是字符串常量那么len(s)也是常量。如果表达式s的类型是array或者array的指针,且表达式不是channel的接收操作或是函数调用,那么len(s)是常量,且表达式s不会被求值;否则len和cap会对s进行求值,其计算结果也不是一个常量。

其实说的很清楚了,但还有三点需要说明。

第一个是视为常量的表达式里为什么不能含有chan的接收操作和函数调用?

这个答案很简单,因为这两个操作都是使用这明确希望发生“副作用”的。特别是从chan里接收数据,还会导致goroutine阻塞,而我们的常量len表达式不会进行求值,这些你期望会发生的副作用便不会产生,会引发一些隐蔽的bug。

第二个是我们注意到了函数调用前用non-constant修饰了,这是什么意思?

按字面意思,一部分函数调用其实是可以在编译期完成计算被当成常量处理的,而另一些不可以。

在进一步深入之前我们先要看看golang里哪些东西是常量/常量表达式。

  • 首先是各种字面量以及对字面量的类型转换产生的值了,无需多说。
  • 一部分内置函数:len、cap、imag、real、complex,它们在参数是常量的时候本身也是常量表达式。
  • unsafe.Sizeof,因为类型的大小也是编译期就能确定的,所以它是常量表达式也很好理解。
  • 所有的常量之间的运算(加减乘除位运算等,除了非常量表达式函数的调用)都是常量表达式。

从上面的描述里可以看出两点,内置函数和unsafe.Sizeof的调用我们可以看成是constant function calls,所有常量表达式除了浮点数和复数表达式都可以在编译期完成计算。而其他函数比如用户自定义函数的调用,虽然仍然有可能在编译期被求值优化,但本身不属于常量表达式。所以语言标准会加以区分。比如下面这个:


func add(x, y int) int {
    return x + y
}

func main() {
    fmt.Println(add(512, 513)) // 1025
}

如果我们看生成的汇编,会发现求值已经完成,不需要调用add:

MOVQ    $1025, (SP)
PCDATA  $1, $0
CALL    runtime.convT64(SB)
MOVQ    8(SP), AX
XORPS   X0, X0
MOVUPS  X0, ""..autotmp_16+64(SP)
LEAQ    type.int(SB), CX
MOVQ    CX, ""..autotmp_16+64(SP)
MOVQ    AX, ""..autotmp_16+72(SP)
NOP
MOVQ    os.Stdout(SB), AX
LEAQ    go.itab.*os.File,io.Writer(SB), CX
MOVQ    CX, (SP)
MOVQ    AX, 8(SP)
LEAQ    ""..autotmp_16+64(SP), AX
MOVQ    AX, 16(SP)
MOVQ    $1, 24(SP)
MOVQ    $1, 32(SP)
NOP
CALL    fmt.Fprintln(SB)
MOVQ    80(SP), BP
ADDQ    $88, SP
RET

很明显的,1025已经在编译期求值了,然而add的调用不是常量表达式,所以下面的代码会报错:


const number = add(512, 513) // error!!!

// example.go:11:7: const initializer add(512, 513) is not a constant

spec给出的实例是调用的内置函数,内置函数也只有在参数是常量的情况下被调用才算做常量表达式:


const (
 c4 = len([10]float64{imag(2i)})  // imag(2i) is a constant and no function call is issued
 c5 = len([10]float64{imag(z)})   // invalid: imag(z) is a (non-constant) function call
)
var z complex128

所以len的表达式里如果用了non-constant的函数调用,那么就len本身不能算是常量表达式了。

这就有了最后一个疑问,题目中的x不是常量,为什么len的结果是常量呢?

标准只说表达式里不能有chan的接收和非常量表达式的函数调用,没说其他的不可以。你也可以这么理解,表达式都有结果值,任何值除了无类型常量(这里显然不是)都是要有一个确定的类型的,只要这个类型是数组或者数组的指针,那len就能获得数组的长度,而这一切不需要s一定是常量表达式,编译器可以简单推导出表达式的值的类型。不能包含non-constant function calls和chan接收是我在第一点里解释的,杜绝所有可能的副作用发生从而保证即使不对表达式求值程序也是正确的,不包含这两个操作的表达式既可以是常量的也可以不是,所以这里我们能用x.s[99]作为len的参数。

说了这么多,只要len的参数类型为array或者array的指针并且符合要求,就不会进行求值,而题目里的表达式正好满足这点,所以虽然我们看起来是会导致panic的代码,但是本身并未进行实际求值,因此程序可以正常运行。另外cap也遵循同样的规则。

最后,还有个小测验,检验一下自己的学习吧:


// 以下哪些语句是正确的,哪些是错误的
var slice [][]*[10]int

const (
    a = len(slice[10000000000000][4]) // 1
    b = len(slice[1]) // 2
    c = len(slice) // 3
    d = len([1]int{1024}) // 4
    e = len([1]int{add(512, 512)}) // 5
    g = len([unsafe.Sizeof(slice)]int{}) // 6
    g = len([unsafe.Sizeof(slice)]int{int(unsafe.Sizeof(slice))}) // 7
)

参考
https://golang.org/ref/speC#Length_and_capacity

到此这篇关于golang内置函数len的小技巧的文章就介绍到这了,更多相关golang内置函数len内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

您可能感兴趣的文档:

--结束END--

本文标题: golang内置函数len的小技巧

本文链接: https://www.lsjlt.com/news/131060.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • golang内置函数len的小技巧
    len是很常用的内置函数,可以测量字符串、slice、array、channel以及map的长度/元素个数。 不过你真的了解len吗?也许还有一些你不知道的小知识。 我们来看一道GO...
    99+
    2022-11-12
  • Golang函数的打印调试技巧
    Golang(又称Go语言)是一门由谷歌开发的,现代化的编程语言,广泛在云计算、大数据、后端开发等领域使用。在开发过程中,程序的正确性和性能往往是开发者最关心的问题之一。因此,在开发中适时地添加打印调试信息对于程序的开发和调试至关重要。本文...
    99+
    2023-05-19
    函数 Golang 调试技巧
  • Golang函数的指针参数使用技巧
    Golang函数的指针参数使用技巧在Golang中,函数的指针参数是非常常见的用法。它可以让我们在函数内部修改传入的参数值,而不需要返回多个值。本文将介绍一些使用指针参数的技巧,以及注意事项。一、指针参数的定义在Golang中,使用指针作为...
    99+
    2023-05-16
    技巧 Golang 函数指针
  • Golang函数的启动goroutine实现技巧
    Golang是一种著名的开发语言,它在近年来在并发编程方面得到了广泛的应用。在Golang中,goroutine是一种轻量级线程,可以轻松地创建和销毁。启动goroutine是一种实现并发编程的常见方式,本文将讨论Golang函数的启动go...
    99+
    2023-05-16
    函数 Golang 启动goroutine
  • Golang函数的字节流处理技巧
    Golang是一种使用广泛的编程语言,它具有高效、可靠和简单易学等特点。在进行一些网络编程和数据处理的时候,字节流处理是十分常见的需求。因此,本文将探讨如何在Golang中实现字节流处理的技巧。一、字节流的概念字节流是计算机中常用的一种数据...
    99+
    2023-05-17
    函数 Golang 字节流处理
  • Golang函数的安全释放锁的技巧
    Golang是一门高效、支持并发的编程语言,常用于开发服务器端应用和云计算平台。在并发编程中,锁是一种常用的机制,用于保护共享资源不被多个并发线程同时访问,从而防止数据竞争和内存泄漏等问题的发生。然而,在使用锁的过程中,需要注意锁的释放问题...
    99+
    2023-05-16
    函数 Golang
  • Golang函数的优雅退出和循环遍历跳出小技巧
    Golang作为一门开发效率高、性能优异的编程语言,其强大的函数功能是其关键特性之一。在开发过程中,经常会遇到需要退出函数或循环遍历的情况。本文将介绍Golang函数的优雅退出和循环遍历跳出小技巧。一、函数的优雅退出在Golang编程中,有...
    99+
    2023-05-16
    Golang函数 优雅退出 循环遍历
  • Golang函数的map类型参数应用技巧
    Golang是一门高效、安全、易于维护的编程语言,越来越多的开发者开始使用它。在Go语言中,函数参数可以是任何类型,包括map类型。本文将介绍一些Golang函数中使用map类型参数的技巧,帮助开发者更好地利用Golang的功能特性进行开发...
    99+
    2023-05-17
    函数 Golang map类型
  • Golang函数的单元测试技巧讲解
    Golang作为一门新兴的语言,已经被越来越多的企业或开发者所采用,而在这些应用场景中,函数作为最基础模块,其稳定性和可靠性是应用成功的关键因素。因此,在进行Golang开发的过程中,单元测试是非常重要的,而这里,我们就来讲解一下Golan...
    99+
    2023-05-16
    技巧 Golang 单元测试
  • Golang函数的指针数组、数组指针的应用技巧
    随着Golang语言的不断发展,函数的指针数组和数组指针也成为了Golang编程中的重要组成部分。在本篇文章中,我们将会探讨Golang函数的指针数组和数组指针的应用技巧,并为大家提供一些实用的示例代码和相关的解释。Golang指针数组的基...
    99+
    2023-05-16
    Golang函数 指针数组 数组指针
  • Golang函数的错误处理和日志记录技巧
    在Golang编程中,函数的错误处理和日志记录至关重要。良好的错误处理和日志记录可以让程序在运行过程中更加健壮和可靠。本文将介绍Golang函数的错误处理和日志记录技巧。一、Golang错误处理在Golang中,错误是一个满足了Go程序包中...
    99+
    2023-05-18
    Golang 错误处理 日志记录
  • Golang函数的结构体类型的嵌入应用技巧
    Golang函数的结构体类型的嵌入应用技巧Golang是一门强类型的编程语言,它支持"对象"的封装,也就是结构体类型的定义。在结构体类型里面也可以使用嵌入类型来扩展已有的类型。在Golang中,嵌入类型实际上就是利用一个类型的名称来作为另一...
    99+
    2023-05-17
    Golang 结构体类型 嵌入应用
  • Golang函数的二进制和十六进制数据处理技巧
    首先,为了更好地理解Golang函数对二进制和十六进制数据的处理技巧,我们需要先明确一下二进制和十六进制的基础知识。二进制,也叫做基数为2的进制,是计算机系统内部最常用的数字表示方式。在二进制中,每个位上的值只有0和1两种可能,例如1011...
    99+
    2023-05-17
    Golang 二进制数据处理 十六进制数据处理
  • Golang函数的原子操作和通道缓冲的使用技巧
    Golang是一门高效且容易编写可扩展软件的编程语言。它具有并发性和并行性,允许程序员以一种简单而直接的方式来编写高性能软件。Golang中的原子操作和通道缓冲是最常用的工具之一,旨在优化程序的性能和稳定性。原子操作是指一种可以保证在多个并...
    99+
    2023-05-17
    Golang 原子操作 通道缓冲
  • Golang函数的文件缓冲和网络缓冲的应用技巧
    Golang作为一种快速高效的编程语言,其函数提供了文件缓冲和网络缓冲的应用技巧。文件缓冲是在文件操作中使用的缓冲技巧,通过减少读写文件的次数来提高性能。网络缓冲则是网络通信中的一种必要技巧,可以提高网络传输的效率。下面将详细介绍这两种技巧...
    99+
    2023-05-18
    Golang 函数 缓冲。
  • Golang函数的字符串处理和字符编码转换技巧
    Golang作为一门编程语言,它所提供的字符串处理和字符编码转换功能非常强大和丰富。本文将介绍Golang函数中常用的字符串处理和字符编码转换技巧,帮助读者更好地理解和使用Golang。一、字符串处理字符串连接在Golang中,可以使用"+...
    99+
    2023-05-18
    字符串处理 Golang函数 字符编码转换技巧
  • 使用WordPress内置图片仓库制作缩略图的小技巧
    WordPress 不仅是博客, 很多时候 WordPress 还被用作为 cms (内容管理系统). 博主们喜欢为每个文章加上统一大小的缩略图, 尤其是信息类平台. 其中比较常用的处理办法是用 custom field...
    99+
    2022-06-12
    图片仓库 缩略图
  • Golang函数的系统调用和文件系统操作的应用技巧
    随着计算机技术的不断发展,各种语言也应运而生。其中,Golang(又称GO语言)因为其高效、简单、易于学习的特点,在近年来越来越受到开发者们的青睐。在Golang中,函数的系统调用和文件系统操作是常见的应用技巧。本文将详细介绍这些技巧的应用...
    99+
    2023-05-17
    Golang函数 文件系统操作 系统调用
  • golang给函数参数设置默认值的几种方式小结(函数参数默认值
    目录前言强制改变使用可变参数语法糖利用结构体的config转换函数的全部参数补充知识:Golang中设置函数默认参数的优雅实现总结前言 这个问题相当麻烦,根据golang-nuts/...
    99+
    2023-01-06
    golang 参数默认值 golang函数参数设置默认值 golang函数定义
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作