Commit a50ee009 authored by Rob Pike's avatar Rob Pike

tutorial: describe unidirectional channels

R=golang-dev, adg, gri
CC=golang-dev
https://golang.org/cl/5370058
parent d9897096
...@@ -1377,7 +1377,7 @@ The <code>server</code> routine loops forever, receiving requests and, to avoid ...@@ -1377,7 +1377,7 @@ The <code>server</code> routine loops forever, receiving requests and, to avoid
a long-running operation, starting a goroutine to do the actual work. a long-running operation, starting a goroutine to do the actual work.
<p> <p>
<pre><!--{{code "progs/server.go" `/func.server/` `/^}/`}} <pre><!--{{code "progs/server.go" `/func.server/` `/^}/`}}
-->func server(op binOp, service chan *request) { -->func server(op binOp, service &lt;-chan *request) {
for { for {
req := &lt;-service req := &lt;-service
go run(op, req) // don&#39;t wait for it go run(op, req) // don&#39;t wait for it
...@@ -1385,17 +1385,41 @@ a long-running operation, starting a goroutine to do the actual work. ...@@ -1385,17 +1385,41 @@ a long-running operation, starting a goroutine to do the actual work.
} }
</pre> </pre>
<p> <p>
We construct a server in a familiar way, starting it and returning a channel There's a new feature in the signature of <code>server</code>: the type of the
<code>service</code> channel specifies the direction of communication.
A channel of plain <code>chan</code> type can be used both for sending and receiving.
However, the type used when declaring a channel can be decorated with an arrow to
indicate that the channel can be used only to send (<code>chan&lt;-</code>) or to
receive (<code>&lt;-chan</code>) data.
The arrow points towards or away from the <code>chan</code> to indicate whether data flows into or out of
the channel.
In the <code>server</code> function, <code>service &lt;-chan *request</code> is a "receive only" channel
that the function can use only to <em>read</em> new requests.
<p>
We instantiate a server in a familiar way, starting it and returning a channel
connected to it: connected to it:
<p> <p>
<pre><!--{{code "progs/server.go" `/func.startServer/` `/^}/`}} <pre><!--{{code "progs/server.go" `/func.startServer/` `/^}/`}}
-->func startServer(op binOp) chan *request { -->func startServer(op binOp) chan&lt;- *request {
req := make(chan *request) req := make(chan *request)
go server(op, req) go server(op, req)
return req return req
} }
</pre> </pre>
<p> <p>
The returned channel is send only, even though the channel was created bidirectionally.
The read end is passed to <code>server</code>, while the send end is returned
to the caller of <code>startServer</code>, so the two halves of the channel
are distinguished, just as we did in <code>startServer</code>.
<p>
Bidirectional channels can be assigned to unidirectional channels but not the
other way around, so if you annotate your channel directions when you declare
them, such as in function signatures, the type system can help you set up and
use channels correctly.
Note that it's pointless to <code>make</code> unidirectional channels, since you can't
use them to communicate. Their purpose is served by variables assigned from bidirectional channels
to distinguish the input and output halves.
<p>
Here's a simple test. It starts a server with an addition operator and sends out Here's a simple test. It starts a server with an addition operator and sends out
<code>N</code> requests without waiting for the replies. Only after all the requests are sent <code>N</code> requests without waiting for the replies. Only after all the requests are sent
does it check the results. does it check the results.
...@@ -1437,7 +1461,7 @@ we can provide a second, <code>quit</code> channel to the server: ...@@ -1437,7 +1461,7 @@ we can provide a second, <code>quit</code> channel to the server:
It passes the quit channel to the <code>server</code> function, which uses it like this: It passes the quit channel to the <code>server</code> function, which uses it like this:
<p> <p>
<pre><!--{{code "progs/server1.go" `/func.server/` `/^}/`}} <pre><!--{{code "progs/server1.go" `/func.server/` `/^}/`}}
-->func server(op binOp, service chan *request, quit chan bool) { -->func server(op binOp, service &lt;-chan *request, quit &lt;-chan bool) {
for { for {
select { select {
case req := &lt;-service: case req := &lt;-service:
......
...@@ -968,11 +968,35 @@ a long-running operation, starting a goroutine to do the actual work. ...@@ -968,11 +968,35 @@ a long-running operation, starting a goroutine to do the actual work.
<p> <p>
{{code "progs/server.go" `/func.server/` `/^}/`}} {{code "progs/server.go" `/func.server/` `/^}/`}}
<p> <p>
We construct a server in a familiar way, starting it and returning a channel There's a new feature in the signature of <code>server</code>: the type of the
<code>service</code> channel specifies the direction of communication.
A channel of plain <code>chan</code> type can be used both for sending and receiving.
However, the type used when declaring a channel can be decorated with an arrow to
indicate that the channel can be used only to send (<code>chan&lt;-</code>) or to
receive (<code>&lt;-chan</code>) data.
The arrow points towards or away from the <code>chan</code> to indicate whether data flows into or out of
the channel.
In the <code>server</code> function, <code>service &lt;-chan *request</code> is a "receive only" channel
that the function can use only to <em>read</em> new requests.
<p>
We instantiate a server in a familiar way, starting it and returning a channel
connected to it: connected to it:
<p> <p>
{{code "progs/server.go" `/func.startServer/` `/^}/`}} {{code "progs/server.go" `/func.startServer/` `/^}/`}}
<p> <p>
The returned channel is send only, even though the channel was created bidirectionally.
The read end is passed to <code>server</code>, while the send end is returned
to the caller of <code>startServer</code>, so the two halves of the channel
are distinguished, just as we did in <code>startServer</code>.
<p>
Bidirectional channels can be assigned to unidirectional channels but not the
other way around, so if you annotate your channel directions when you declare
them, such as in function signatures, the type system can help you set up and
use channels correctly.
Note that it's pointless to <code>make</code> unidirectional channels, since you can't
use them to communicate. Their purpose is served by variables assigned from bidirectional channels
to distinguish the input and output halves.
<p>
Here's a simple test. It starts a server with an addition operator and sends out Here's a simple test. It starts a server with an addition operator and sends out
<code>N</code> requests without waiting for the replies. Only after all the requests are sent <code>N</code> requests without waiting for the replies. Only after all the requests are sent
does it check the results. does it check the results.
......
...@@ -18,14 +18,14 @@ func run(op binOp, req *request) { ...@@ -18,14 +18,14 @@ func run(op binOp, req *request) {
req.replyc <- reply req.replyc <- reply
} }
func server(op binOp, service chan *request) { func server(op binOp, service <-chan *request) {
for { for {
req := <-service req := <-service
go run(op, req) // don't wait for it go run(op, req) // don't wait for it
} }
} }
func startServer(op binOp) chan *request { func startServer(op binOp) chan<- *request {
req := make(chan *request) req := make(chan *request)
go server(op, req) go server(op, req)
return req return req
......
...@@ -18,7 +18,7 @@ func run(op binOp, req *request) { ...@@ -18,7 +18,7 @@ func run(op binOp, req *request) {
req.replyc <- reply req.replyc <- reply
} }
func server(op binOp, service chan *request, quit chan bool) { func server(op binOp, service <-chan *request, quit <-chan bool) {
for { for {
select { select {
case req := <-service: case req := <-service:
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment