说说 Go 语言的坑(二)

简介: 上一篇文章说的是 for-range 的,工作中,其实还是遇到蛮多奇奇怪怪的问题,这里也顺便整理了一下,就当作是续集:)

上一篇文章 说说 Go 语言 for-range 的坑 说的是 for-range 的,工作中,其实还是遇到蛮多奇奇怪怪的问题,这里也顺便整理了一下,就当作是续集:)

先继续看 for-range 的另一个坑:

下面代码输出什么?

func main() {
   
    var a = []int{
   1, 2, 3, 4, 5}
    var r = make([]int, 0)

    for i, v := range a {
   
        if i == 0 {
   
            a = append(a, 6, 7)
        }
        r = append(r, v)
    }
    fmt.Println(r)
}

答案:[1 2 3 4 5]。
剖析:for range 后跟的都是值拷贝。所以 a 在遍历的时候,新增了 6 和 7 两个元素,但是 range a 是使用 a 的副本参与循环,副本的 len = 5,所以 r = [1 2 3 4 5],也就是只获取到 a 的底层数组的前 5 个元素。

OK,相信你看懂了。那这道题呢?

func main() {
   
    var a = [5]int{
   1, 2, 3, 4, 5}
    var r [5]int
    for i, v := range a {
   
        if i == 0 {
   
            a[1] = 12
            a[2] = 13
        }
        r[i] = v
    }
    fmt.Println("r = ", r)
    fmt.Println("a = ", a)
}

答案:r=[1 2 3 4 5],a=[1 12 13 4 5]。
剖析:原理同上,range a 此时会复制数组 a,但如果 a 的定义为切片,var a = []int{1, 2, 3, 4, 5},最终的输出,r 和 a 就是一样的了,都是 [1 12 13 4 5]。

最后一行输出什么?

func main() {
   
    x := 1
    fmt.Println(x)
    {
   
        fmt.Println(x)
        i, x := 2, 2
        fmt.Println(i, x)
    }
    fmt.Println(x) // print ?
}

答案:1。
知识点:变量隐藏。
剖析:使用变量简短声明符号 := 时,符号左边多个变量,只需保证至少有一个变量是新声明的,并对已定义的变量尽进行赋值操作。但如果出现作用域,会导致变量隐藏的问题。如果你使用 Goland,你会发现 x 变量,变成了另一种颜色,也就是说这两个变量不一样,因为作用域不一样。
小结:非常常见,又十分隐秘的坑。

下面代码输出什么?

func main() {
   
    x := interface{
   }(nil)
    _, y := x.(interface{
   })
    println(y)
}

答案:false。
知识点:类型断言。
剖析:

  1. 类型断言语法:i.(Type),其中 i 是接口,Type 是类型或接口。
  2. 编译时,编译器会自动检测 i 的动态类型与 Type 是否一致。但是,如果动态类型不存在,则断言总是失败——本例 x 没有类型,所以输出 false 。

下面代码是否正确?

func main() {
   
    m := make(map[string]int, 2)
    cap(m)
}

答案:false。
剖析:

  1. 使用 make 创建 map 变量时可以指定第二个参数,不过会被忽略。
  2. cap 函数适用于数组、数组指针、slice 和 channel,不适用于 map。

位运算的坑整理

  1. 取反符号是 ^,而不是 ~,这点跟很多语言不一样。
  2. ^ 是二元运算符的时候,代表异或运算。
  3. &^ 这是 按位置零 符号,看个例子就明白了:
func main() {
   
    var x uint8 = 214
    var y uint8 = 92
    fmt.Printf("x: %08b\n", x)
    fmt.Printf("y: %08b\n", y)
    fmt.Printf("x | y: %08b\n", x|y)
    fmt.Printf("x &^ y: %08b\n", x&^y)
}

程序输出:

x: 11010110
y: 01011100
x | y: 11011110
x &^ y: 10000010

也就是说,y 的哪一位为 1,输出的那一位就是 0,和 | 运算相反。这个符号其他语言好像没见过,看了例子,是不是觉得也挺简单的。


文章来源于本人博客,发布于 2019-08-25,原文链接:https://imlht.com/archives/192/

目录
相关文章
|
2月前
|
存储 监控 算法
员工上网行为监控中的Go语言算法:布隆过滤器的应用
在信息化高速发展的时代,企业上网行为监管至关重要。布隆过滤器作为一种高效、节省空间的概率性数据结构,适用于大规模URL查询与匹配,是实现精准上网行为管理的理想选择。本文探讨了布隆过滤器的原理及其优缺点,并展示了如何使用Go语言实现该算法,以提升企业网络管理效率和安全性。尽管存在误报等局限性,但合理配置下,布隆过滤器为企业提供了经济有效的解决方案。
85 8
员工上网行为监控中的Go语言算法:布隆过滤器的应用
|
2月前
|
存储 Go 索引
go语言中数组和切片
go语言中数组和切片
47 7
|
2月前
|
Go 开发工具
百炼-千问模型通过openai接口构建assistant 等 go语言
由于阿里百炼平台通义千问大模型没有完善的go语言兼容openapi示例,并且官方答复assistant是不兼容openapi sdk的。 实际使用中发现是能够支持的,所以自己写了一个demo test示例,给大家做一个参考。
|
2月前
|
程序员 Go
go语言中结构体(Struct)
go语言中结构体(Struct)
116 71
|
2月前
|
存储 Go 索引
go语言中的数组(Array)
go语言中的数组(Array)
117 67
|
10天前
|
存储 监控 算法
内网监控系统之 Go 语言布隆过滤器算法深度剖析
在数字化时代,内网监控系统对企业和组织的信息安全至关重要。布隆过滤器(Bloom Filter)作为一种高效的数据结构,能够快速判断元素是否存在于集合中,适用于内网监控中的恶意IP和违规域名筛选。本文介绍其原理、优势及Go语言实现,提升系统性能与响应速度,保障信息安全。
22 5
|
2月前
|
Go 索引
go语言for遍历数组或切片
go语言for遍历数组或切片
119 62
|
19天前
|
算法 安全 Go
Go语言中的加密和解密是如何实现的?
Go语言通过标准库中的`crypto`包提供丰富的加密和解密功能,包括对称加密(如AES)、非对称加密(如RSA、ECDSA)及散列函数(如SHA256)。`encoding/base64`包则用于Base64编码与解码。开发者可根据需求选择合适的算法和密钥,使用这些包进行加密操作。示例代码展示了如何使用`crypto/aes`包实现对称加密。加密和解密操作涉及敏感数据处理,需格外注意安全性。
38 14
|
19天前
|
Go 数据库
Go语言中的包(package)是如何组织的?
在Go语言中,包是代码组织和管理的基本单元,用于集合相关函数、类型和变量,便于复用和维护。包通过目录结构、文件命名、初始化函数(`init`)及导出规则来管理命名空间和依赖关系。合理的包组织能提高代码的可读性、可维护性和可复用性,减少耦合度。例如,`stringutils`包提供字符串处理函数,主程序导入使用这些函数,使代码结构清晰易懂。
63 11
|
19天前
|
存储 安全 Go
Go语言中的map数据结构是如何实现的?
Go 语言中的 `map` 是基于哈希表实现的键值对数据结构,支持快速查找、插入和删除操作。其原理涉及哈希函数、桶(Bucket)、动态扩容和哈希冲突处理等关键机制,平均时间复杂度为 O(1)。为了确保线程安全,Go 提供了 `sync.Map` 类型,通过分段锁实现并发访问的安全性。示例代码展示了如何使用自定义结构体和切片模拟 `map` 功能,以及如何使用 `sync.Map` 进行线程安全的操作。