gopl-zh.github.com/ch8/ch8-04.md

49 lines
2.5 KiB
Markdown
Raw Normal View History

2016-01-02 14:07:09 +00:00
## 8.4. Channels
2016-07-20 08:17:54 +00:00
如果说goroutine是Go语言程序的并发体的话那么channels则是它们之间的通信机制。一个channels是一个通信机制它可以让一个goroutine通过它给另一个goroutine发送值信息。每个channel都有一个特殊的类型也就是channels可发送数据的类型。一个可以发送int类型数据的channel一般写为chan int。
2016-01-03 04:07:46 +00:00
2016-02-15 03:06:34 +00:00
使用内置的make函数我们可以创建一个channel
2016-01-03 04:07:46 +00:00
```Go
ch := make(chan int) // ch has type 'chan int'
```
2016-09-30 13:56:19 +00:00
和map类似channel也一个对应make创建的底层数据结构的引用。当我们复制一个channel或用于函数参数传递时我们只是拷贝了一个channel引用因此调用者和被调用者将引用同一个channel对象。和其它的引用类型一样channel的零值也是nil。
2016-01-03 04:07:46 +00:00
2016-09-30 13:56:19 +00:00
两个相同类型的channel可以使用==运算符比较。如果两个channel引用的是相同的对象那么比较的结果为真。一个channel也可以和nil进行比较。
2016-01-03 04:07:46 +00:00
2016-02-15 03:06:34 +00:00
一个channel有发送和接受两个主要操作都是通信行为。一个发送语句将一个值从一个goroutine通过channel发送到另一个执行接收操作的goroutine。发送和接收两个操作都是用`<-`运算符在发送语句中`<-`运算符分割channel和要发送的值在接收语句中`<-`运算符写在channel对象之前一个不使用接收结果的接收操作也是合法的
2016-01-03 04:07:46 +00:00
```Go
ch <- x // a send statement
x = <-ch // a receive expression in an assignment statement
<-ch // a receive statement; result is discarded
```
2016-09-30 13:56:19 +00:00
Channel还支持close操作用于关闭channel随后对基于该channel的任何发送操作都将导致panic异常。对一个已经被close过的channel进行接收操作依然可以接受到之前已经成功发送的数据如果channel中已经没有数据的话讲产生一个零值的数据。
2016-01-03 04:07:46 +00:00
2016-02-15 03:06:34 +00:00
使用内置的close函数就可以关闭一个channel
2016-01-03 04:07:46 +00:00
```Go
close(ch)
```
2016-02-15 03:06:34 +00:00
以最简单方式调用make函数创建的时一个无缓存的channel但是我们也可以指定第二个整形参数对应channel的容量。如果channel的容量大于零那么该channel就是带缓存的channel。
2016-01-03 04:07:46 +00:00
```Go
ch = make(chan int) // unbuffered channel
ch = make(chan int, 0) // unbuffered channel
ch = make(chan int, 3) // buffered channel with capacity 3
```
2016-02-15 03:06:34 +00:00
我们将先讨论无缓存的channel然后在8.4.4节讨论带缓存的channel。
2016-01-03 04:07:46 +00:00
2016-01-02 14:07:09 +00:00
{% include "./ch8-04-1.md" %}
{% include "./ch8-04-2.md" %}
{% include "./ch8-04-3.md" %}
{% include "./ch8-04-4.md" %}