hincky的主页 hincky的主页
  • 学习笔记

    • Vue笔记
    • Vuepress
    • nginx
  • 语言类

    • java
    • go
    • python
    • 设计模式
  • 框架类

    • Spring
    • Spring Security
    • Mybatis
  • 容器技术

    • docker
    • k8s
    • helm
    • prometheus
    • grafana
    • jenkins
  • 命令集合

    • linux命令
    • docker命令
    • git命令
    • vim命令
    • k8s命令
  • 数据库

    • sql
    • mysql
  • 协议

    • 网络模型
    • http/1.1
    • WebSocket
    • http/2
    • TLS/SSL
    • tcp
    • IP
    • tcpdump抓包命令
    • wireshark抓包工具
  • 通用

    • Git
  • 技术分享

    • git push/pull总是超时怎么办
    • idea debug技巧
    • postman使用
    • 问题总结
    • idea使用技巧
  • Oauth2

    • Oauth2原理
  • 项目列表

    • redis项目
    • 微服务项目
  • 分类
  • 标签
  • 归档
  • 随笔
GitHub (opens new window)

Hincky

当有趣的人,做想做的事
  • 学习笔记

    • Vue笔记
    • Vuepress
    • nginx
  • 语言类

    • java
    • go
    • python
    • 设计模式
  • 框架类

    • Spring
    • Spring Security
    • Mybatis
  • 容器技术

    • docker
    • k8s
    • helm
    • prometheus
    • grafana
    • jenkins
  • 命令集合

    • linux命令
    • docker命令
    • git命令
    • vim命令
    • k8s命令
  • 数据库

    • sql
    • mysql
  • 协议

    • 网络模型
    • http/1.1
    • WebSocket
    • http/2
    • TLS/SSL
    • tcp
    • IP
    • tcpdump抓包命令
    • wireshark抓包工具
  • 通用

    • Git
  • 技术分享

    • git push/pull总是超时怎么办
    • idea debug技巧
    • postman使用
    • 问题总结
    • idea使用技巧
  • Oauth2

    • Oauth2原理
  • 项目列表

    • redis项目
    • 微服务项目
  • 分类
  • 标签
  • 归档
  • 随笔
GitHub (opens new window)
  • java

  • python

  • Spring

  • SpringMVC

  • SpringSecurity

  • Mybatis

  • 设计模式

  • Go

    • 基础语法

      • 环境安装
        • 你好,Go语言
        • 环境安装
        • 目录结构
        • 命令
        • 开发工具
        • 学习网址
      • 变量声明
        • 概述
        • 数据类型
          • 字符串
          • 布尔
          • 数字
        • 常量声明
        • 变量声明
        • 输出方法
      • 数组
        • 概述
        • 声明数组
        • 注意事项
      • Slice 切片
        • 概述
        • 声明切片
        • 截取切片
        • 追加切片
        • 删除切片
      • Struct 结构体
        • 概述
        • 声明结构体
        • 生成 JSON
        • 改变数据
      • Map 集合
        • 概述
        • 声明 Map
        • 生成 JSON
        • 编辑和删除
      • 循环
        • 概述
        • 循环 array
        • 循环 slice
        • 循环 map
        • break
        • continue
        • goto
        • switch
      • 函数
        • 概述
        • 函数定义
        • 值传递
        • 引用传递
        • MD5
        • 获取当前时间字符串
        • 获取当前时间戳
        • 生成签名
      • chan 通道
        • 概述
        • 声明 chan
        • 写入 chan
        • 读取 chan
        • 关闭 chan
        • 示例
      • defer 函数
        • 概述
        • 执行顺序
        • 闭包
        • Return
          • 一
          • 二
          • 三
          • 四
        • os.Exit
        • 不同协程
        • 答案解析
        • go-gin-api 系列文章
      • 解析 JSON 数据
        • 概述
        • JSON 结构
        • go-gin-api 系列文章
      • Unmarshal 遇到的小坑
        • 1.问题现象描述
        • 2.问题影响描述
        • 3.引起问题的原因
        • 4.问题的解决方案
      • 结构(struct) 实现 接口(interface)
        • 代码示例
        • 代码解释
          • 一、
          • 二、
          • 三、
      • Dial(target string, opts …DialOption) 的写法
        • 一
        • 二
        • 场景
        • 代码实现
        • 输出
      • RFC3339 时间格式化
        • 运行一下
        • 小结
      • 常用签名算法的基准测试
        • MD5 单向散列加密
        • AES 对称加密
        • RSA 非对称加密
        • 最后
      • 分享两个在开发中需注意的小点
        • 不要使用 + 和 fmt.Sprintf 操作字符串
          • +
          • fmt.Sprintf
          • bytes.NewBufferString
        • 对于固定字段的键值对,不要使用 map[string]interface{}
          • map[string]interface{}
          • 临时 Struct
        • 小结
          • 推荐阅读
      • Pool 来减少 GC 压力
        • 前言
        • sync.Pool
        • 小结
        • 推荐阅读
      • 基于逃逸分析来提升程序性能
        • 前言
        • 什么是逃逸分析?
        • 如何确定是否逃逸?
        • 可能出现逃逸的场景
          • 01
          • 02
          • 03
        • 小结
        • 推荐阅读
      • Map 来解决 map 的并发操作问题
        • 前言
        • map 并发操作出现问题
        • sync.Map 解决并发操作问题
        • 计算 map 长度
        • 计算 sync.Map 长度
        • 小结
        • 推荐阅读
      • WaitGroup 来实现并发操作
        • 前言
        • sync.WaitGroup 正确使用
        • sync.WaitGroup 闭坑指南
          • 01
          • 02
          • 03
        • 小结
        • 推荐阅读
    • Gin框架

    • Go gRPC

    • go-gin-api [文档]

    • go基本介绍
      • 安装go
      • hello world
      • 变量类型
      • 变量和声明
        • 定义和赋值分开
        • 定义和赋值合并
        • 最简单的写法
        • 最简写法结合函数
        • 多个变量同时赋值
        • 变量总结
      • if
      • 导入包
      • 运行go代码
      • 函数声明
        • 返回值个数
        • 参数接收
        • 两个参数都接收
        • 只接收其中一个参数
    • Go结构体
      • 定义结构体
      • 声明和初始化
        • 结构体字段
        • 结构体初始化
        • 函数参数传递方式
        • 镜像复制
        • 指针
      • 结构体上的函数
      • 构造器
        • new
    • Go相关项目推荐
      • 收录golang的一些好的开源项目
  • 后端
  • Go
  • 基础语法
hincky
2022-11-21
目录

chan 通道

# 概述

原来分享基础语法的时候,还未分享过 chan 通道,这次把它补上。

chan 可以理解为队列,遵循先进先出的规则。

在说 chan 之前,咱们先说一下 go 关键字。

在 go 关键字后面加一个函数,就可以创建一个线程,函数可以为已经写好的函数,也可以是匿名函数。

举个例子:

func main() {
	fmt.Println("main start")

	go func() {
		fmt.Println("goroutine")
	}()

	fmt.Println("main end")
}
1
2
3
4
5
6
7
8
9

输出:

main start
main end
1
2

为什么没有输出 goroutine ?

首先,我们清楚 Go 语言的线程是并发机制,不是并行机制。

那么,什么是并发,什么是并行?

并发是不同的代码块交替执行,也就是交替可以做不同的事情。

并行是不同的代码块同时执行,也就是同时可以做不同的事情。

举个生活化场景的例子:

你正在家看书,忽然电话来了,然后你接电话,通话完成后继续看书,这就是并发,看书和接电话交替做。

如果电话来了,你一边看书一遍接电话,这就是并行,看书和接电话一起做。

说回上面的例子,为什么没有输出 goroutine ?

main 函数是一个主线程,是因为主线程执行太快了,子线程还没来得及执行,所以看不到输出。

现在让主线程休眠 1 秒钟,再试试。

func main() {
	fmt.Println("main start")

	go func() {
		fmt.Println("goroutine")
	}()

	time.Sleep(1 * time.Second)

	fmt.Println("main end")
}
1
2
3
4
5
6
7
8
9
10
11

输出:

main start
goroutine
main end
1
2
3

这就对了。

接下来,看看如何使用 chan 。

# 声明 chan

// 声明不带缓冲的通道
ch1 := make(chan string)

// 声明带10个缓冲的通道
ch2 := make(chan string, 10)

// 声明只读通道
ch3 := make(<-chan string)

// 声明只写通道
ch4 := make(chan<- string)
1
2
3
4
5
6
7
8
9
10
11

注意:

不带缓冲的通道,进和出都会阻塞。

带缓冲的通道,进一次长度 +1,出一次长度 -1,如果长度等于缓冲长度时,再进就会阻塞。

# 写入 chan

ch1 := make(chan string, 10)

ch1 <- "a"
1
2
3

# 读取 chan

val, ok := <- ch1
// 或
val := <- ch1
1
2
3

# 关闭 chan

close(chan)
1

注意:

  • close 以后不能再写入,写入会出现 panic
  • 重复 close 会出现 panic
  • 只读的 chan 不能 close
  • close 以后还可以读取数据

# 示例

func main() {
	fmt.Println("main start")
	ch := make(chan string)
	ch <- "a" // 入 chan
	go func() {
		val := <- ch // 出 chan
		fmt.Println(val)
	}()
	fmt.Println("main end")
}
1
2
3
4
5
6
7
8
9
10

输出:

main start
fatal error: all goroutines are asleep - deadlock!
1
2

What ? 这是为啥,刚开始就出师不利呀?

因为,定义的是一个无缓冲的 chan,赋值后就陷入了阻塞。

怎么解决它?

声明一个有缓冲的 chan。

func main() {
	fmt.Println("main start")
	ch := make(chan string, 1)
	ch <- "a" // 入 chan
	go func() {
		val := <- ch // 出 chan
		fmt.Println(val)
	}()
	fmt.Println("main end")
}
1
2
3
4
5
6
7
8
9
10

输出:

main start
main end
1
2

为啥没有输出 a , 和前面一样,主线程执行太快了,加个休眠 1 秒钟,再试试。

func main() {
	fmt.Println("main start")
	ch := make(chan string, 1)
	ch <- "a" // 入 chan
	go func() {
		val := <- ch // 出 chan
		fmt.Println(val)
	}()
	time.Sleep(1 * time.Second)
	fmt.Println("main end")
}
1
2
3
4
5
6
7
8
9
10
11

输出:

main start
a
main end
1
2
3

这就对了。

再看一个例子:

func main() {
	fmt.Println("main start")
	ch := make(chan string)
	go func() {
		ch <- "a" // 入 chan
	}()
	go func() {
		val := <- ch // 出 chan
		fmt.Println(val)
	}()
	time.Sleep(1 * time.Second)
	fmt.Println("main end")
}
1
2
3
4
5
6
7
8
9
10
11
12
13

输出:

main start
a
main end
1
2
3

再看一个例子:

func producer(ch chan string) {
	fmt.Println("producer start")
	ch <- "a"
	ch <- "b"
	ch <- "c"
	ch <- "d"
	fmt.Println("producer end")
}

func main() {
	fmt.Println("main start")
	ch := make(chan string, 3)
	go producer(ch)

	time.Sleep(1 * time.Second)
	fmt.Println("main end")
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17

输出:

main start
producer start
main end
1
2
3

带缓冲的通道,如果长度等于缓冲长度时,再进就会阻塞。

再看一个例子:

func producer(ch chan string) {
	fmt.Println("producer start")
	ch <- "a"
	ch <- "b"
	ch <- "c"
	ch <- "d"
	fmt.Println("producer end")
}

func customer(ch chan string) {
	for {
		msg := <- ch
		fmt.Println(msg)
	}
}

func main() {
	fmt.Println("main start")
	ch := make(chan string, 3)
	go producer(ch)
	go customer(ch)

	time.Sleep(1 * time.Second)
	fmt.Println("main end")
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25

输出:

main start
producer start
producer end
a
b
c
d
main end
1
2
3
4
5
6
7
8

就到这吧。

编辑 (opens new window)
#Go
函数
defer 函数

← 函数 defer 函数→

最近更新
01
人生前期重要的能力
05-17
02
防火墙命令
04-11
03
docker-compose部署mysql主从集群
03-22
更多文章>
Theme by Vdoing | Copyright © 2022-2023 Hincky | MIT License | 粤ICP备2022120427号
  • 跟随系统
  • 浅色模式
  • 深色模式
  • 阅读模式