Golang随笔之《Go专家编程》查漏补缺

前言

本文为书籍阅读笔记,查漏补缺.
相关底层剖析Golang底层原理剖析专栏汇总

第一章 常见数据结构的实现原理

管道

普通使用场景

  1. 读写nil管道均会阻塞,而且是永久阻塞
  2. 关闭的管道仍然可以读取数据
  3. 向关闭的管道写数据会触发panic

v, ok := <-ch的ok含义

package main

import "fmt"

func main() {
    
    
	ch := make(chan int, 2)
	ch <- 6
	ch <- 8
	close(ch)
	for i := 0; ; i++ {
    
    
		v, ok := <-ch
		if ok {
    
    
			fmt.Println("第", i, "第读取成功,value=", v, "ok=true")
		} else {
    
    
			fmt.Println("第", i, "第读取失败,", "ok=false")
			break
		}
	}
}
0 第读取成功,value= 6 ok=true
第 1 第读取成功,value= 8 ok=true
第 2 第读取失败, ok=false

v, ok := <-ch
第一个变量表示读出的数据,第二个变量bool类型,表示是否成功读取了数据,需要注意的是,第二个变量不用于指示管道关闭的状态!需要注意的是,第二个变量不用于指示管道关闭的状态
管道已关闭但是缓冲区中仍有数据,那么管道读取表达返回的第一个变量为读取到的数据,第二个变量为true。可以看到,只有管道已关闭且缓冲区中没有数据时,管道读取表达式返回的第二个变量才跟关闭状态一致。

读写channel底层逻辑图例

在这里插入图片描述
向管道写数据在时在实现有一个小技巧,当接收队列recvq不为空时,说明缓冲区中没有数但是有协程在等待数据,此时会把数据直接传递给recvq队列中的第一个协程,而不必再写入缓冲区
在这里插入图片描述
类似的,如果等待发送队列sendq不为空,且没有缓冲区,那么此时将直接从sendq队列的第一个协程中获取数据

slice

简单表达式:

  • b:=a[low:height]
  • len(b)=height-low
  • cap(b)=cap(a)-low

扩展表达式:

  • b:=a[low:height:max]
  • len(b)=height-low
  • cap(b)=max-low

解释

在使用简单表达式的时候,用append函数增加新的元素时,可能会覆盖之前切片的元素,所以我们需要一种限制新切片容量的表达式,即扩展表达式,我们称扩展表达式为被封印的切片,当使用append函数向被封印的切片追加新元素时,如果容量不足则会产生一个全新的切片,而不会覆盖原始的数组或切片

iota

本质

iota代表了const声明块的行索引,从0开始

特点:如果为常量指定了一个表达式,但后续的常量没有表达式,则会继续上面的表达式

string

双引号和反单引号的区别

hi
this is wxf

使用双引号表达时,需要对特殊字符转义

s:="hi,\nthis is \"wxf\""

使用反单引号时,不需要对特殊字符转义

s:=`hi,
this is wxf`

string的注意点

  1. string可能为空,但不会是nil
  2. 字符串拼接时会触发内存分配以及内存拷贝(s=s+“wxf”)
  3. 无论是string转[]byte还是[]byte转string,都会发生一次内存拷贝,会有一定的开销
  4. 字符串的长度是指字节数,而非字符数

第二章 控制结构

select

复习点

直接看一遍之前的博文即可,书上的不是很详细
Golang底层原理剖析之多路select、channel数据结构和阻塞与非阻塞

for-range

range不同类型的区别

  • channel:遇到nil的channel会永久阻塞
  • slice:遍历slice或者array时已经决定了循环次数,append也没事,新添加的数据不会被遍历到
  • map:遍历map时不要添加或者删除元素,会发生不可遇到的情况
  • channel:range通道时,只有当close(ch)后,遍历完了range才会自动退出,如果没有close,那么就阻塞等待

Guess you like

Origin blog.csdn.net/qq_42956653/article/details/121523065