This commit is contained in:
chai2010
2015-12-14 12:08:47 +08:00
parent 3b8de6c274
commit 0a24303905
132 changed files with 1069 additions and 159 deletions

View File

@@ -48,7 +48,7 @@
<body>
<div class="book" data-level="8.1" data-chapter-title="Goroutines" data-filepath="ch8/ch8-01.md" data-basepath=".." data-revision="Wed Dec 09 2015 15:54:13 GMT+0800 (中国标准时间)">
<div class="book" data-level="8.1" data-chapter-title="Goroutines" data-filepath="ch8/ch8-01.md" data-basepath=".." data-revision="Mon Dec 14 2015 11:30:54 GMT+0800 (中国标准时间)">
<div class="book-summary">

View File

@@ -48,7 +48,7 @@
<body>
<div class="book" data-level="8.2" data-chapter-title="示例: 併髮的Clock服務" data-filepath="ch8/ch8-02.md" data-basepath=".." data-revision="Wed Dec 09 2015 15:54:13 GMT+0800 (中国标准时间)">
<div class="book" data-level="8.2" data-chapter-title="示例: 併髮的Clock服務" data-filepath="ch8/ch8-02.md" data-basepath=".." data-revision="Mon Dec 14 2015 11:30:54 GMT+0800 (中国标准时间)">
<div class="book-summary">

View File

@@ -48,7 +48,7 @@
<body>
<div class="book" data-level="8.3" data-chapter-title="示例: 併髮的Echo服務" data-filepath="ch8/ch8-03.md" data-basepath=".." data-revision="Wed Dec 09 2015 15:54:13 GMT+0800 (中国标准时间)">
<div class="book" data-level="8.3" data-chapter-title="示例: 併髮的Echo服務" data-filepath="ch8/ch8-03.md" data-basepath=".." data-revision="Mon Dec 14 2015 11:30:54 GMT+0800 (中国标准时间)">
<div class="book-summary">

View File

@@ -48,7 +48,7 @@
<body>
<div class="book" data-level="8.4" data-chapter-title="Channels" data-filepath="ch8/ch8-04.md" data-basepath=".." data-revision="Wed Dec 09 2015 15:54:13 GMT+0800 (中国标准时间)">
<div class="book" data-level="8.4" data-chapter-title="Channels" data-filepath="ch8/ch8-04.md" data-basepath=".." data-revision="Mon Dec 14 2015 11:30:54 GMT+0800 (中国标准时间)">
<div class="book-summary">

View File

@@ -48,7 +48,7 @@
<body>
<div class="book" data-level="8.5" data-chapter-title="併行的循環" data-filepath="ch8/ch8-05.md" data-basepath=".." data-revision="Wed Dec 09 2015 15:54:13 GMT+0800 (中国标准时间)">
<div class="book" data-level="8.5" data-chapter-title="併行的循環" data-filepath="ch8/ch8-05.md" data-basepath=".." data-revision="Mon Dec 14 2015 11:30:54 GMT+0800 (中国标准时间)">
<div class="book-summary">

View File

@@ -48,7 +48,7 @@
<body>
<div class="book" data-level="8.6" data-chapter-title="示例: 併髮的Web爬蟲" data-filepath="ch8/ch8-06.md" data-basepath=".." data-revision="Wed Dec 09 2015 15:54:13 GMT+0800 (中国标准时间)">
<div class="book" data-level="8.6" data-chapter-title="示例: 併髮的Web爬蟲" data-filepath="ch8/ch8-06.md" data-basepath=".." data-revision="Mon Dec 14 2015 11:30:54 GMT+0800 (中国标准时间)">
<div class="book-summary">
@@ -2060,7 +2060,139 @@
<section class="normal" id="section-">
<h2 id="86-&#x793A;&#x4F8B;-&#x4F75;&#x9AEE;&#x7684;web&#x722C;&#x87F2;">8.6. &#x793A;&#x4F8B;: &#x4F75;&#x9AEE;&#x7684;Web&#x722C;&#x87F2;</h2>
<p>TODO</p>
<p>&#x5728;5.6&#x7BC0;&#x4E2D;&#xFF0C;&#x6211;&#x5011;&#x505A;&#x4E86;&#x4E00;&#x7B87;&#x7C21;&#x55AE;&#x7684;web&#x722C;&#x87F2;&#xFF0C;&#x7528;bfs(&#x5EE3;&#x5EA6;&#x512A;&#x5148;)&#x7B97;&#x6CD5;&#x4F86;&#x6293;&#x53D6;&#x6574;&#x7B87;&#x7DB2;&#x7AD9;&#x3002;&#x5728;&#x672C;&#x7BC0;&#x4E2D;&#xFF0C;&#x6211;&#x5011;&#x6703;&#x8B93;&#x9019;&#x7B87;&#x9019;&#x7B87;&#x722C;&#x87F2;&#x4F75;&#x884C;&#x5316;&#xFF0C;&#x9019;&#x6A23;&#x6BCF;&#x4E00;&#x7B87;&#x5F7C;&#x6B64;&#x7368;&#x7ACB;&#x7684;&#x6293;&#x53D6;&#x547D;&#x4EE4;&#x53EF;&#x4EE5;&#x4F75;&#x884C;&#x9032;&#x884C;IO&#xFF0C;&#x6700;&#x5927;&#x5316;&#x5229;&#x7528;&#x7DB2;&#x7D61;&#x8CC7;&#x6E90;&#x3002;crawl&#x51FD;&#x6578;&#x548C;gopl.io/ch5/findlinks3&#x4E2D;&#x7684;&#x662F;&#x4E00;&#x6A23;&#x7684;&#x3002;</p>
<pre><code class="lang-go">gopl.io/ch8/crawl1
<span class="hljs-keyword">func</span> crawl(url <span class="hljs-typename">string</span>) []<span class="hljs-typename">string</span> {
fmt.Println(url)
list, err := links.Extract(url)
<span class="hljs-keyword">if</span> err != <span class="hljs-constant">nil</span> {
log.Print(err)
}
<span class="hljs-keyword">return</span> list
}
</code></pre>
<p>&#x4E3B;&#x51FD;&#x6578;&#x548C;5.6&#x7BC0;&#x4E2D;&#x7684;breadthFirst(&#x6DF1;&#x5EA6;&#x512A;&#x5148;)&#x985E;&#x4F3C;&#x3002;&#x50CF;&#x4E4B;&#x524D;&#x4E00;&#x6A23;&#xFF0C;&#x4E00;&#x7B87;worklist&#x662F;&#x4E00;&#x7B87;&#x8A18;&#x9332;&#x4E86;&#x9700;&#x8981;&#x8655;&#x7406;&#x7684;&#x5143;&#x7D20;&#x7684;&#x968A;&#x5217;&#xFF0C;&#x6BCF;&#x4E00;&#x7B87;&#x5143;&#x7D20;&#x90FD;&#x662F;&#x4E00;&#x7B87;&#x9700;&#x8981;&#x6293;&#x53D6;&#x7684;URL&#x5217;&#x9336;&#xFF0C;&#x4E0D;&#x904E;&#x9019;&#x4E00;&#x6B21;&#x6211;&#x5011;&#x7528;channel&#x4EE3;&#x66FF;slice&#x4F86;&#x505A;&#x9019;&#x7B87;&#x968A;&#x5217;&#x3002;&#x6BCF;&#x4E00;&#x7B87;&#x5C0D;crawl&#x7684;&#x8ABF;&#x7528;&#x90FD;&#x6703;&#x5728;&#x4ED6;&#x5011;&#x81EA;&#x5DF1;&#x7684;goroutine&#x4E2D;&#x9032;&#x884C;&#x4F75;&#x4E14;&#x6703;&#x628A;&#x4ED6;&#x5011;&#x6293;&#x5230;&#x7684;&#x93C8;&#x63A5;&#x9AEE;&#x9001;&#x8FF4;worklist&#x3002;</p>
<pre><code class="lang-go"><span class="hljs-keyword">func</span> main() {
worklist := <span class="hljs-built_in">make</span>(<span class="hljs-keyword">chan</span> []<span class="hljs-typename">string</span>)
<span class="hljs-comment">// Start with the command-line arguments.</span>
<span class="hljs-keyword">go</span> <span class="hljs-keyword">func</span>() { worklist &lt;- os.Args[<span class="hljs-number">1</span>:] }()
<span class="hljs-comment">// Crawl the web concurrently.</span>
seen := <span class="hljs-built_in">make</span>(<span class="hljs-keyword">map</span>[<span class="hljs-typename">string</span>]<span class="hljs-typename">bool</span>)
<span class="hljs-keyword">for</span> list := <span class="hljs-keyword">range</span> worklist {
<span class="hljs-keyword">for</span> _, link := <span class="hljs-keyword">range</span> list {
<span class="hljs-keyword">if</span> !seen[link] {
seen[link] = <span class="hljs-constant">true</span>
<span class="hljs-keyword">go</span> <span class="hljs-keyword">func</span>(link <span class="hljs-typename">string</span>) {
worklist &lt;- crawl(link)
}(link)
}
}
}
}
</code></pre>
<p>&#x8A3B;&#x610F;&#x9019;&#x88CF;&#x7684;crawl&#x6240;&#x5728;&#x7684;goroutine&#x6703;&#x5C07;link&#x4F5C;&#x70BA;&#x4E00;&#x7B87;&#x986F;&#x5F0F;&#x7684;&#x8518;&#x6578;&#x50B3;&#x5165;&#xFF0C;&#x4F86;&#x907F;&#x514D;&#x201C;&#x5FAA;&#x74B0;&#x8B8A;&#x91CF;&#x5FEB;&#x7167;&#x201D;&#x7684;&#x554F;&#x984C;(&#x5728;5.6.1&#x4E2D;&#x6709;&#x8B1B;&#x89E3;)&#x3002;&#x53E6;&#x5916;&#x8A3B;&#x610F;&#x9019;&#x88CF;&#x5C07;&#x547D;&#x4EE4;&#x884C;&#x8518;&#x6578;&#x50B3;&#x5165;worklist&#x4E5F;&#x662F;&#x5728;&#x4E00;&#x7B87;&#x53E6;&#x5916;&#x7684;goroutine&#x4E2D;&#x9032;&#x884C;&#x7684;&#xFF0C;&#x9019;&#x662F;&#x70BA;&#x4E86;&#x907F;&#x514D;&#x5728;main goroutine&#x548C;crawler goroutine&#x4E2D;&#x8855;&#x6642;&#x56AE;&#x53E6;&#x4E00;&#x7B87;goroutine&#x901A;&#x904E;channel&#x9AEE;&#x9001;&#x5167;&#x5BB9;&#x6642;&#x9AEE;&#x751F;&#x6B7B;&#x9396;(&#x56E0;&#x70BA;&#x53E6;&#x4E00;&#x908A;&#x7684;&#x63A5;&#x6536;&#x64CD;&#x4F5C;&#x9084;&#x6C92;&#x6709;&#x6E96;&#x5099;&#x597D;)&#x3002;&#x5679;&#x7136;&#xFF0C;&#x9019;&#x88CF;&#x6211;&#x5011;&#x4E5F;&#x53EF;&#x4EE5;&#x7528;buffered channel&#x4F86;&#x89E3;&#x6C7A;&#x554F;&#x984C;&#xFF0C;&#x9019;&#x88CF;&#x4E0D;&#x518D;&#x8D05;&#x8FF0;&#x3002;</p>
<p>&#x73FE;&#x5728;&#x722C;&#x87F2;&#x53EF;&#x4EE5;&#x9AD8;&#x4F75;&#x9AEE;&#x5730;&#x904B;&#x884C;&#x8D77;&#x4F86;&#xFF0C;&#x4F75;&#x4E14;&#x53EF;&#x4EE5;&#x7522;&#x751F;&#x4E00;&#x5927;&#x5768;&#x7684;URL&#x4E86;&#xFF0C;&#x4E0D;&#x904E;&#x9084;&#x662F;&#x6703;&#x6709;&#x5006;&#x554F;&#x984C;&#x3002;&#x4E00;&#x7B87;&#x554F;&#x984C;&#x662F;&#x5728;&#x904B;&#x884C;&#x4E00;&#x6BB5;&#x6642;&#x9593;&#x5F8C;&#x53EF;&#x80FD;&#x6703;&#x9F63;&#x73FE;&#x5728;log&#x7684;&#x932F;&#x8AA4;&#x4FE1;&#x606F;&#x88CF;&#x7684;&#xFF1A;</p>
<pre><code>$ go build gopl.io/ch8/crawl1
$ ./crawl1 http://gopl.io/
http://gopl.io/
https://golang.org/help/
https://golang.org/doc/
https://golang.org/blog/
...
2015/07/15 18:22:12 Get ...: dial tcp: lookup blog.golang.org: no such host
2015/07/15 18:22:12 Get ...: dial tcp 23.21.222.120:443: socket:
too many open files
...
</code></pre><p>&#x6700;&#x521D;&#x7684;&#x932F;&#x8AA4;&#x4FE1;&#x606F;&#x662F;&#x4E00;&#x7B87;&#x8B93;&#x4EBA;&#x83AB;&#x540D;&#x7684;DNS&#x67FB;&#x627E;&#x5931;&#x6557;&#xFF0C;&#x537D;&#x4F7F;&#x9019;&#x7B87;&#x57DF;&#x540D;&#x662F;&#x5B8C;&#x5168;&#x53EF;&#x9760;&#x7684;&#x3002;&#x800C;&#x96A8;&#x5F8C;&#x7684;&#x932F;&#x8AA4;&#x4FE1;&#x606F;&#x63ED;&#x793A;&#x4E86;&#x539F;&#x56E0;&#xFF1A;&#x9019;&#x7B87;&#x7A0B;&#x5E8F;&#x4E00;&#x6B21;&#x6027;&#x5275;&#x5EFA;&#x4E86;&#x592A;&#x591A;&#x7DB2;&#x7D61;&#x9023;&#x63A5;&#xFF0C;&#x8D85;&#x904E;&#x4E86;&#x6BCF;&#x4E00;&#x7B87;&#x9032;&#x7A0B;&#x7684;&#x6253;&#x958B;&#x6587;&#x4EF6;&#x6578;&#x9650;&#x88FD;&#xFF0C;&#x65E3;&#x800C;&#x5C0E;&#x7DFB;&#x4E86;&#x5728;&#x8ABF;&#x7528;net.Dial&#x50CF;DNS&#x67FB;&#x627E;&#x5931;&#x6557;&#x9019;&#x6A23;&#x7684;&#x554F;&#x984C;&#x3002;</p>
<p>&#x9019;&#x7B87;&#x7A0B;&#x5E8F;&#x5BE6;&#x5728;&#x662F;&#x592A;&#x4ED6;&#x5ABD;&#x4F75;&#x884C;&#x4E86;&#x3002;&#x7121;&#x7AAE;&#x7121;&#x76E1;&#x5730;&#x4F75;&#x884C;&#x5316;&#x4F75;&#x4E0D;&#x662F;&#x4EC0;&#x9EBC;&#x597D;&#x4E8B;&#x60C5;&#xFF0C;&#x56E0;&#x70BA;&#x4E0D;&#x7BA1;&#x600E;&#x9EBC;&#x8AAC;&#xFF0C;&#x4F60;&#x7684;&#x4FC2;&#x7D71;&#x7E3D;&#x662F;&#x6703;&#x6709;&#x4E00;&#x7B87;&#x4E9B;&#x9650;&#x88FD;&#x56E0;&#x7D20;&#xFF0C;&#x6BD4;&#x5982;CPU&#x8988;&#x5FC3;&#x6578;&#x6703;&#x9650;&#x88FD;&#x4F60;&#x7684;&#x8A08;&#x7B97;&#x8CA0;&#x8F09;&#xFF0C;&#x6BD4;&#x5982;&#x4F60;&#x7684;&#x786C;&#x76E4;&#x8F49;&#x8EF8;&#x548C;&#x78C1;&#x982D;&#x6578;&#x9650;&#x88FD;&#x4E86;&#x4F60;&#x7684;&#x672C;&#x5730;&#x78C1;&#x76E4;IO&#x64CD;&#x4F5C;&#x983B;&#x7387;&#xFF0C;&#x6BD4;&#x5982;&#x4F60;&#x7684;&#x7DB2;&#x7D61;&#x5E36;&#x5BEC;&#x9650;&#x88FD;&#x4E86;&#x4F60;&#x7684;&#x4E0B;&#x8F09;&#x901F;&#x5EA6;&#x4E0A;&#x9650;&#xFF0C;&#x6216;&#x8005;&#x662F;&#x4F60;&#x7684;&#x4E00;&#x7B87;web&#x670D;&#x52D9;&#x7684;&#x670D;&#x52D9;&#x5BB9;&#x91CF;&#x4E0A;&#x9650;&#x7B49;&#x7B49;&#x3002;&#x70BA;&#x4E86;&#x89E3;&#x6C7A;&#x9019;&#x7B87;&#x554F;&#x984C;&#xFF0C;&#x6211;&#x5011;&#x53EF;&#x4EE5;&#x9650;&#x88FD;&#x4F75;&#x9AEE;&#x7A0B;&#x5E8F;&#x6240;&#x4F7F;&#x7528;&#x7684;&#x8CC7;&#x6E90;&#x4F86;&#x4F7F;&#x4E4B;&#x9069;&#x61C9;&#x81EA;&#x5DF1;&#x7684;&#x904B;&#x884C;&#x74B0;&#x5883;&#x3002;&#x5C0D;&#x65BC;&#x6211;&#x5011;&#x7684;&#x4F8B;&#x5B50;&#x4F86;&#x8AAC;&#xFF0C;&#x6700;&#x7C21;&#x55AE;&#x7684;&#x65B9;&#x6CD5;&#x5C31;&#x662F;&#x9650;&#x88FD;&#x5C0D;links.Extract&#x5728;&#x8855;&#x4E00;&#x6642;&#x9593;&#x6700;&#x591A;&#x4E0D;&#x6703;&#x6709;&#x8D85;&#x904E;n&#x6B21;&#x8ABF;&#x7528;&#xFF0C;&#x9019;&#x88CF;&#x7684;n&#x662F;fd&#x7684;limit-20&#xFF0C;&#x4E00;&#x822C;&#x60C5;&#x6CC1;&#x4E0B;&#x3002;&#x9019;&#x7B87;&#x4E00;&#x7B87;&#x591C;&#x5E97;&#x88CF;&#x9650;&#x88FD;&#x5BA2;&#x4EBA;&#x6578;&#x76EE;&#x662F;&#x4E00;&#x7B87;&#x9053;&#x7406;&#xFF0C;&#x96BB;&#x6709;&#x5679;&#x6709;&#x5BA2;&#x4EBA;&#x96E2;&#x958B;&#x6642;&#xFF0C;&#x7E94;&#x6703;&#x5141;&#x8A31;&#x65B0;&#x7684;&#x5BA2;&#x4EBA;&#x9032;&#x5165;&#x5E97;&#x5167;(&#x8B6F;&#x8A3B;&#xFF1A;&#x4F5C;&#x8005;&#x4F60;&#x7B87;&#x8001;&#x6D41;&#x6C13;)&#x3002;</p>
<p>&#x6211;&#x5011;&#x53EF;&#x4EE5;&#x7528;&#x4E00;&#x7B87;&#x6709;&#x5BB9;&#x91CF;&#x9650;&#x88FD;&#x7684;buffered channel&#x4F86;&#x63A7;&#x88FD;&#x4F75;&#x9AEE;&#xFF0C;&#x9019;&#x985E;&#x4F3C;&#x65BC;&#x64CD;&#x4F5C;&#x4FC2;&#x7D71;&#x88CF;&#x7684;&#x8A08;&#x6578;&#x4FE1;&#x865F;&#x91CF;&#x6982;&#x5538;&#x3002;&#x5F9E;&#x6982;&#x5538;&#x4E0A;&#x8B1B;&#xFF0C;channel&#x88CF;&#x7684;n&#x7B87;&#x7A7A;&#x69FD;&#x4EE3;&#x9336;n&#x7B87;&#x53EF;&#x4EE5;&#x8655;&#x7406;&#x5167;&#x5BB9;&#x7684;token(&#x901A;&#x884C;&#x8A3C;)&#xFF0C;&#x5F9E;channel&#x88CF;&#x63A5;&#x6536;&#x4E00;&#x7B87;&#x503C;&#x6703;&#x91CB;&#x653E;&#x5176;&#x4E2D;&#x7684;&#x4E00;&#x7B87;token&#xFF0C;&#x4F75;&#x4E14;&#x751F;&#x6210;&#x4E00;&#x7B87;&#x65B0;&#x7684;&#x7A7A;&#x69FD;&#x4F4D;&#x3002;&#x9019;&#x6A23;&#x4FDD;&#x8A3C;&#x4E86;&#x5728;&#x6C92;&#x6709;&#x63A5;&#x6536;&#x4ECB;&#x5165;&#x6642;&#x6700;&#x591A;&#x6709;n&#x7B87;&#x9AEE;&#x9001;&#x64CD;&#x4F5C;&#x3002;(&#x9019;&#x88CF;&#x53EF;&#x80FD;&#x6211;&#x5011;&#x62FF;channel&#x88CF;&#x586B;&#x5145;&#x7684;&#x69FD;&#x4F86;&#x505A;token&#x66F4;&#x76F4;&#x89C0;&#x4E00;&#x4E9B;&#xFF0C;&#x4E0D;&#x904E;&#x9084;&#x662F;&#x9019;&#x6A23;&#x5427;~)&#x3002;&#x7531;&#x65BC;channel&#x88CF;&#x7684;&#x5143;&#x7D20;&#x985E;&#x578B;&#x4F75;&#x4E0D;&#x91CD;&#x8981;&#xFF0C;&#x6211;&#x5011;&#x7528;&#x4E00;&#x7B87;&#x96F6;&#x503C;&#x7684;struct{}&#x4F86;&#x4F5C;&#x70BA;&#x5176;&#x5143;&#x7D20;&#x3002;</p>
<p>&#x8B93;&#x6211;&#x5011;&#x91CD;&#x5BEB;crawl&#x51FD;&#x6578;&#xFF0C;&#x5C07;&#x5C0D;links.Extract&#x7684;&#x8ABF;&#x7528;&#x64CD;&#x4F5C;&#x7528;&#x7A6B;&#x53D6;&#x3001;&#x91CB;&#x653E;token&#x7684;&#x64CD;&#x4F5C;&#x5305;&#x88F9;&#x8D77;&#x4F86;&#xFF0C;&#x4F86;&#x78BA;&#x4FDD;&#x8855;&#x4E00;&#x6642;&#x9593;&#x5C0D;&#x5176;&#x96BB;&#x6709;20&#x7B87;&#x8ABF;&#x7528;&#x3002;&#x4FE1;&#x865F;&#x91CF;&#x6578;&#x91CF;&#x548C;&#x5176;&#x80FD;&#x64CD;&#x4F5C;&#x7684;IO&#x8CC7;&#x6E90;&#x6578;&#x91CF;&#x61C9;&#x4FDD;&#x6301;&#x63A5;&#x8FD1;&#x3002;</p>
<pre><code class="lang-go">gopl.io/ch8/crawl2
<span class="hljs-comment">// tokens is a counting semaphore used to</span>
<span class="hljs-comment">// enforce a limit of 20 concurrent requests.</span>
<span class="hljs-keyword">var</span> tokens = <span class="hljs-built_in">make</span>(<span class="hljs-keyword">chan</span> <span class="hljs-keyword">struct</span>{}, <span class="hljs-number">20</span>)
<span class="hljs-keyword">func</span> crawl(url <span class="hljs-typename">string</span>) []<span class="hljs-typename">string</span> {
fmt.Println(url)
tokens &lt;- <span class="hljs-keyword">struct</span>{}{} <span class="hljs-comment">// acquire a token</span>
list, err := links.Extract(url)
&lt;-tokens <span class="hljs-comment">// release the token</span>
<span class="hljs-keyword">if</span> err != <span class="hljs-constant">nil</span> {
log.Print(err)
}
<span class="hljs-keyword">return</span> list
}
</code></pre>
<p>&#x7B2C;&#x4E8C;&#x500B;&#x554F;&#x984C;&#x662F;&#x9019;&#x500B;&#x7A0B;&#x5E8F;&#x6C38;&#x9060;&#x90FD;&#x4E0D;&#x6703;&#x7D42;&#x6B62;&#xFF0C;&#x537D;&#x4F7F;&#x5B83;&#x5DF2;&#x7D93;&#x722C;&#x5230;&#x4E86;&#x6240;&#x6709;&#x521D;&#x59CB;&#x93C8;&#x63A5;&#x884D;&#x751F;&#x9F63;&#x7684;&#x93C8;&#x63A5;&#x3002;(&#x5679;&#x7136;&#xFF0C;&#x9664;&#x975E;&#x4F60;&#x614E;&#x91CD;&#x5730;&#x9078;&#x64C7;&#x4E86;&#x95A4;&#x9069;&#x7684;&#x521D;&#x59CB;&#x5316;URL&#x6216;&#x8005;&#x5DF2;&#x7D93;&#x5BE6;&#x73FE;&#x4E86;&#x7DF4;&#x7FD2;8.6&#x4E2D;&#x7684;&#x6DF1;&#x5EA6;&#x9650;&#x88FD;&#xFF0C;&#x4F60;&#x61C9;&#x8A72;&#x9084;&#x6C92;&#x6709;&#x610F;&#x8B58;&#x5230;&#x9019;&#x500B;&#x554F;&#x984C;)&#x3002;&#x7232;&#x4E86;&#x4F7F;&#x9019;&#x500B;&#x7A0B;&#x5E8F;&#x80FD;&#x5920;&#x7D42;&#x6B62;&#xFF0C;&#x6211;&#x5011;&#x9700;&#x8981;&#x5728;worklist&#x7232;&#x7A7A;&#x6216;&#x8005;&#x6C92;&#x6709;crawl&#x7684;goroutine&#x5728;&#x904B;&#x884C;&#x6642;&#x9000;&#x9F63;&#x4E3B;&#x5FAA;&#x74B0;&#x3002;</p>
<pre><code class="lang-go"><span class="hljs-keyword">func</span> main() {
worklist := <span class="hljs-built_in">make</span>(<span class="hljs-keyword">chan</span> []<span class="hljs-typename">string</span>)
<span class="hljs-keyword">var</span> n <span class="hljs-typename">int</span> <span class="hljs-comment">// number of pending sends to worklist</span>
<span class="hljs-comment">// Start with the command-line arguments.</span>
n++
<span class="hljs-keyword">go</span> <span class="hljs-keyword">func</span>() { worklist &lt;- os.Args[<span class="hljs-number">1</span>:] }()
<span class="hljs-comment">// Crawl the web concurrently.</span>
seen := <span class="hljs-built_in">make</span>(<span class="hljs-keyword">map</span>[<span class="hljs-typename">string</span>]<span class="hljs-typename">bool</span>)
<span class="hljs-keyword">for</span> ; n &gt; <span class="hljs-number">0</span>; n-- {
list := &lt;-worklist
<span class="hljs-keyword">for</span> _, link := <span class="hljs-keyword">range</span> list {
<span class="hljs-keyword">if</span> !seen[link] {
seen[link] = <span class="hljs-constant">true</span>
n++
<span class="hljs-keyword">go</span> <span class="hljs-keyword">func</span>(link <span class="hljs-typename">string</span>) {
worklist &lt;- crawl(link)
}(link)
}
}
}
}
</code></pre>
<p>&#x9019;&#x7B87;&#x7248;&#x672C;&#x4E2D;&#xFF0C;&#x8A08;&#x7B97;&#x5668;n&#x5C0D;worklist&#x7684;&#x9AEE;&#x9001;&#x64CD;&#x4F5C;&#x6578;&#x91CF;&#x9032;&#x884C;&#x4E86;&#x9650;&#x88FD;&#x3002;&#x6BCF;&#x4E00;&#x6B21;&#x6211;&#x5011;&#x9AEE;&#x73FE;&#x6709;&#x5143;&#x7D20;&#x9700;&#x8981;&#x88AB;&#x9AEE;&#x9001;&#x5230;worklist&#x6642;&#xFF0C;&#x6211;&#x5011;&#x90FD;&#x6703;&#x5C0D;n&#x9032;&#x884C;++&#x64CD;&#x4F5C;&#xFF0C;&#x5728;&#x56AE;worklist&#x4E2D;&#x9AEE;&#x9001;&#x521D;&#x59CB;&#x7684;&#x547D;&#x4EE4;&#x884C;&#x8518;&#x6578;&#x4E4B;&#x524D;&#xFF0C;&#x6211;&#x5011;&#x4E5F;&#x9032;&#x884C;&#x904E;&#x4E00;&#x6B21;++&#x64CD;&#x4F5C;&#x3002;&#x9019;&#x88CF;&#x7684;&#x64CD;&#x4F5C;++&#x662F;&#x5728;&#x6BCF;&#x5553;&#x52D5;&#x4E00;&#x7B87;crawler&#x7684;goroutine&#x4E4B;&#x524D;&#x3002;&#x4E3B;&#x5FAA;&#x74B0;&#x6703;&#x5728;n&#x6E1B;&#x70BA;0&#x6642;&#x7D42;&#x6B62;&#xFF0C;&#x9019;&#x6642;&#x5019;&#x8AAC;&#x660E;&#x6C92;&#x6D3B;&#x53EF;&#x4E7E;&#x4E86;&#x3002;</p>
<p>&#x73FE;&#x5728;&#x9019;&#x7B87;&#x4F75;&#x9AEE;&#x722C;&#x87F2;&#x6703;&#x6BD4;5.6&#x7BC0;&#x4E2D;&#x7684;&#x6DF1;&#x5EA6;&#x512A;&#x5148;&#x8490;&#x7D22;&#x7248;&#x5FEB;&#x4E0A;20&#x500D;&#xFF0C;&#x800C;&#x4E14;&#x4E0D;&#x6703;&#x9F63;&#x4EC0;&#x9EBC;&#x932F;&#xFF0C;&#x4F75;&#x4E14;&#x5728;&#x5176;&#x5B8C;&#x6210;&#x4EFB;&#x52D9;&#x6642;&#x4E5F;&#x6703;&#x6B63;&#x78BA;&#x5730;&#x7D42;&#x6B62;&#x3002;</p>
<p>&#x4E0B;&#x9EAA;&#x7684;&#x7A0B;&#x5E8F;&#x662F;&#x907F;&#x514D;&#x904E;&#x5EA6;&#x4F75;&#x9AEE;&#x7684;&#x53E6;&#x4E00;&#x7A2E;&#x601D;&#x8DEF;&#x3002;&#x9019;&#x7B87;&#x7248;&#x672C;&#x4F7F;&#x7528;&#x4E86;&#x539F;&#x4F86;&#x7684;crawl&#x51FD;&#x6578;&#xFF0C;&#x4F46;&#x6C92;&#x6709;&#x4F7F;&#x7528;&#x8A08;&#x6578;&#x4FE1;&#x865F;&#x91CF;&#xFF0C;&#x53D6;&#x800C;&#x4EE3;&#x4E4B;&#x7528;&#x4E86;20&#x7B87;&#x9577;&#x6D3B;&#x7684;crawler goroutine&#xFF0C;&#x9019;&#x6A23;&#x4F86;&#x4FDD;&#x8A3C;&#x6700;&#x591A;20&#x7B87;HTTP&#x8ACB;&#x6C42;&#x5728;&#x4F75;&#x9AEE;&#x3002;</p>
<pre><code class="lang-go"><span class="hljs-keyword">func</span> main() {
worklist := <span class="hljs-built_in">make</span>(<span class="hljs-keyword">chan</span> []<span class="hljs-typename">string</span>) <span class="hljs-comment">// lists of URLs, may have duplicates</span>
unseenLinks := <span class="hljs-built_in">make</span>(<span class="hljs-keyword">chan</span> <span class="hljs-typename">string</span>) <span class="hljs-comment">// de-duplicated URLs</span>
<span class="hljs-comment">// Add command-line arguments to worklist.</span>
<span class="hljs-keyword">go</span> <span class="hljs-keyword">func</span>() { worklist &lt;- os.Args[<span class="hljs-number">1</span>:] }()
<span class="hljs-comment">// Create 20 crawler goroutines to fetch each unseen link.</span>
<span class="hljs-keyword">for</span> i := <span class="hljs-number">0</span>; i &lt; <span class="hljs-number">20</span>; i++ {
<span class="hljs-keyword">go</span> <span class="hljs-keyword">func</span>() {
<span class="hljs-keyword">for</span> link := <span class="hljs-keyword">range</span> unseenLinks {
foundLinks := crawl(link)
<span class="hljs-keyword">go</span> <span class="hljs-keyword">func</span>() { worklist &lt;- foundLinks }()
}
}()
}
<span class="hljs-comment">// The main goroutine de-duplicates worklist items</span>
<span class="hljs-comment">// and sends the unseen ones to the crawlers.</span>
seen := <span class="hljs-built_in">make</span>(<span class="hljs-keyword">map</span>[<span class="hljs-typename">string</span>]<span class="hljs-typename">bool</span>)
<span class="hljs-keyword">for</span> list := <span class="hljs-keyword">range</span> worklist {
<span class="hljs-keyword">for</span> _, link := <span class="hljs-keyword">range</span> list {
<span class="hljs-keyword">if</span> !seen[link] {
seen[link] = <span class="hljs-constant">true</span>
unseenLinks &lt;- link
}
}
}
}
</code></pre>
<p>&#x6240;&#x6709;&#x7684;&#x722C;&#x87F2;goroutine&#x73FE;&#x5728;&#x90FD;&#x662F;&#x88AB;&#x8855;&#x4E00;&#x7B87;channel-unseenLinks&#x9935;&#x98FD;&#x7684;&#x4E86;&#x3002;&#x4E3B;goroutine&#x8CA0;&#x8CAC;&#x62C6;&#x5206;&#x5B83;&#x5F9E;worklist&#x88CF;&#x62FF;&#x5230;&#x7684;&#x5143;&#x7D20;&#xFF0C;&#x7136;&#x5F8C;&#x628A;&#x6C92;&#x6709;&#x6293;&#x904E;&#x7684;&#x7D93;&#x7531;unseenLinks channel&#x9AEE;&#x9001;&#x7D66;&#x4E00;&#x7B87;&#x722C;&#x87F2;&#x7684;goroutine&#x3002;</p>
<p>seen&#x9019;&#x7B87;map&#x88AB;&#x9650;&#x5B9A;&#x5728;main goroutine&#x4E2D;&#xFF1B;&#x4E5F;&#x5C31;&#x662F;&#x8AAC;&#x9019;&#x7B87;map&#x96BB;&#x80FD;&#x5728;main goroutine&#x4E2D;&#x9032;&#x884C;&#x8A2A;&#x554F;&#x3002;&#x985E;&#x4F3C;&#x65BC;&#x5176;&#x5B83;&#x7684;&#x4FE1;&#x606F;&#x96B1;&#x85CF;&#x65B9;&#x5F0F;&#xFF0C;&#x9019;&#x6A23;&#x7684;&#x7D04;&#x675F;&#x53EF;&#x4EE5;&#x8B93;&#x6211;&#x5011;&#x5F9E;&#x4E00;&#x5B9A;&#x7A0B;&#x5EA6;&#x4E0A;&#x4FDD;&#x8A3C;&#x7A0B;&#x5E8F;&#x7684;&#x6B63;&#x78BA;&#x6027;&#x3002;&#x4F8B;&#x5982;&#xFF0C;&#x5167;&#x90E8;&#x8B8A;&#x91CF;&#x4E0D;&#x80FD;&#x5920;&#x5728;&#x51FD;&#x6578;&#x5916;&#x90E8;&#x88AB;&#x8A2A;&#x554F;&#x5230;&#xFF1B;&#x8B8A;&#x91CF;(&#xA7;2.3.4)&#x5728;&#x6C92;&#x6709;&#x88AB;&#x8F49;&#x7FA9;&#x7684;&#x60C5;&#x6CC1;&#x4E0B;&#x662F;&#x7121;&#x6CD5;&#x5728;&#x51FD;&#x6578;&#x5916;&#x90E8;&#x8A2A;&#x554F;&#x7684;&#xFF1B;&#x4E00;&#x7B87;&#x5C0D;&#x8C61;&#x7684;&#x5C01;&#x88DD;&#x5B57;&#x6BB5;&#x7121;&#x6CD5;&#x88AB;&#x8A72;&#x5C0D;&#x8C61;&#x7684;&#x65B9;&#x6CD5;&#x4EE5;&#x5916;&#x7684;&#x65B9;&#x6CD5;&#x8A2A;&#x554F;&#x5230;&#x3002;&#x5728;&#x6240;&#x6709;&#x7684;&#x60C5;&#x6CC1;&#x4E0B;&#xFF0C;&#x4FE1;&#x606F;&#x96B1;&#x85CF;&#x90FD;&#x53EF;&#x4EE5;&#x5E6B;&#x52A9;&#x6211;&#x5011;&#x7D04;&#x675F;&#x6211;&#x5011;&#x7684;&#x7A0B;&#x5E8F;&#xFF0C;&#x4F7F;&#x5176;&#x4E0D;&#x9AEE;&#x751F;&#x610F;&#x6599;&#x4E4B;&#x5916;&#x7684;&#x60C5;&#x6CC1;&#x3002;</p>
<p>crawl&#x51FD;&#x6578;&#x722C;&#x5230;&#x7684;&#x93C8;&#x63A5;&#x5728;&#x4E00;&#x7B87;&#x5C08;&#x6709;&#x7684;goroutine&#x4E2D;&#x88AB;&#x9AEE;&#x9001;&#x5230;worklist&#x4E2D;&#x4F86;&#x907F;&#x514D;&#x6B7B;&#x9396;&#x3002;&#x70BA;&#x4E86;&#x7BC0;&#x7701;&#x7A7A;&#x9593;&#xFF0C;&#x9019;&#x7B87;&#x4F8B;&#x5B50;&#x7684;&#x7D42;&#x6B62;&#x554F;&#x984C;&#x6211;&#x5011;&#x5148;&#x4E0D;&#x9032;&#x884C;&#x8A73;&#x7D30;&#x95E1;&#x8FF0;&#x4E86;&#x3002;</p>
<p>&#x7DF4;&#x7FD2;8.6: &#x70BA;&#x4F75;&#x9AEE;&#x722C;&#x87F2;&#x589E;&#x52A0;&#x6DF1;&#x5EA6;&#x9650;&#x88FD;&#x3002;&#x4E5F;&#x5C31;&#x662F;&#x8AAC;&#xFF0C;&#x5982;&#x679C;&#x7528;&#x6236;&#x8A2D;&#x7F6E;&#x4E86;depth=3&#xFF0C;&#x90A3;&#x9EBC;&#x96BB;&#x6709;&#x5F9E;&#x9996;&#x9801;&#x8DF3;&#x8F49;&#x4E09;&#x6B21;&#x4EE5;&#x5167;&#x80FD;&#x5920;&#x8DF3;&#x5230;&#x7684;&#x9801;&#x9EAA;&#x7E94;&#x80FD;&#x88AB;&#x6293;&#x53D6;&#x5230;&#x3002;</p>
<p>&#x7DF4;&#x7FD2;8.7: &#x5B8C;&#x6210;&#x4E00;&#x7B87;&#x4F75;&#x9AEE;&#x7A0B;&#x5E8F;&#x4F86;&#x5275;&#x5EFA;&#x4E00;&#x7B87;&#x7DDA;&#x4E0A;&#x7DB2;&#x7AD9;&#x7684;&#x672C;&#x5730;&#x93E1;&#x50CF;&#xFF0C;&#x628A;&#x8A72;&#x7AD9;&#x9EDE;&#x7684;&#x6240;&#x6709;&#x53EF;&#x9054;&#x7684;&#x9801;&#x9EAA;&#x90FD;&#x6293;&#x53D6;&#x5230;&#x672C;&#x5730;&#x786C;&#x76E4;&#x3002;&#x70BA;&#x4E86;&#x7701;&#x4E8B;&#xFF0C;&#x6211;&#x5011;&#x9019;&#x88CF;&#x53EF;&#x4EE5;&#x96BB;&#x53D6;&#x9F63;&#x73FE;&#x5728;&#x8A72;&#x57DF;&#x4E0B;&#x7684;&#x6240;&#x6709;&#x9801;&#x9EAA;(&#x6BD4;&#x5982;golang.org&#x7D50;&#x5C3E;&#xFF0C;&#x8B6F;&#x8A3B;&#xFF1A;&#x5916;&#x93C8;&#x7684;&#x61C9;&#x8A72;&#x5C31;&#x4E0D;&#x7B97;&#x4E86;&#x3002;)&#x5679;&#x7136;&#x4E86;&#xFF0C;&#x9F63;&#x73FE;&#x5728;&#x9801;&#x9EAA;&#x88CF;&#x7684;&#x93C8;&#x63A5;&#x4F60;&#x4E5F;&#x9700;&#x8981;&#x9032;&#x884C;&#x4E00;&#x4E9B;&#x8655;&#x7406;&#xFF0C;&#x4F7F;&#x5176;&#x80FD;&#x5920;&#x5728;&#x4F60;&#x7684;&#x93E1;&#x50CF;&#x7AD9;&#x9EDE;&#x4E0A;&#x9032;&#x884C;&#x8DF3;&#x8F49;&#xFF0C;&#x800C;&#x4E0D;&#x662F;&#x6307;&#x56AE;&#x539F;&#x59CB;&#x7684;&#x93C8;&#x63A5;&#x3002;</p>
<p>&#x8B6F;&#x8A3B;&#xFF1A;
&#x62D3;&#x5C55;&#x95B1;&#x8B80;&#xFF1A;
<a href="http://marcio.io/2015/07/handling-1-million-requests-per-minute-with-golang/" target="_blank">http://marcio.io/2015/07/handling-1-million-requests-per-minute-with-golang/</a></p>
</section>

View File

@@ -48,7 +48,7 @@
<body>
<div class="book" data-level="8.7" data-chapter-title="基於select的多路復用" data-filepath="ch8/ch8-07.md" data-basepath=".." data-revision="Wed Dec 09 2015 15:54:13 GMT+0800 (中国标准时间)">
<div class="book" data-level="8.7" data-chapter-title="基於select的多路復用" data-filepath="ch8/ch8-07.md" data-basepath=".." data-revision="Mon Dec 14 2015 11:30:54 GMT+0800 (中国标准时间)">
<div class="book-summary">

View File

@@ -48,7 +48,7 @@
<body>
<div class="book" data-level="8.8" data-chapter-title="示例: 併髮的字典遍歷" data-filepath="ch8/ch8-08.md" data-basepath=".." data-revision="Wed Dec 09 2015 15:54:13 GMT+0800 (中国标准时间)">
<div class="book" data-level="8.8" data-chapter-title="示例: 併髮的字典遍歷" data-filepath="ch8/ch8-08.md" data-basepath=".." data-revision="Mon Dec 14 2015 11:30:54 GMT+0800 (中国标准时间)">
<div class="book-summary">
@@ -2060,7 +2060,160 @@
<section class="normal" id="section-">
<h2 id="88-&#x793A;&#x4F8B;-&#x4F75;&#x9AEE;&#x7684;&#x5B57;&#x5178;&#x904D;&#x6B77;">8.8. &#x793A;&#x4F8B;: &#x4F75;&#x9AEE;&#x7684;&#x5B57;&#x5178;&#x904D;&#x6B77;</h2>
<p>TODO</p>
<p>&#x5728;&#x672C;&#x5C0F;&#x7BC0;&#x4E2D;&#xFF0C;&#x6211;&#x5011;&#x6703;&#x5275;&#x5EFA;&#x4E00;&#x500B;&#x7A0B;&#x5E8F;&#x4F86;&#x751F;&#x6210;&#x6307;&#x5B9A;&#x76EE;&#x5F54;&#x7684;&#x786C;&#x76E4;&#x4F7F;&#x7528;&#x60C5;&#x6CC1;&#x5831;&#x544A;&#xFF0C;&#x9019;&#x500B;&#x7A0B;&#x5E8F;&#x548C;Unix&#x88E1;&#x7684;du&#x5DE5;&#x5177;&#x6BD4;&#x8F03;&#x76F8;&#x4F3C;&#x3002;&#x5927;&#x591A;&#x6578;&#x5DE5;&#x4F5C;&#x7528;&#x4E0B;&#x9EAB;&#x9019;&#x500B;walkDir&#x51FD;&#x6578;&#x4F86;&#x5B8C;&#x6210;&#xFF0C;&#x9019;&#x500B;&#x51FD;&#x6578;&#x4F7F;&#x7528;dirents&#x51FD;&#x6578;&#x4F86;&#x679A;&#x8209;&#x4E00;&#x500B;&#x76EE;&#x5F54;&#x4E0B;&#x7684;&#x6240;&#x6709;&#x5165;&#x53E3;&#x3002;</p>
<pre><code class="lang-go">gopl.io/ch8/du1
<span class="hljs-comment">// walkDir recursively walks the file tree rooted at dir</span>
<span class="hljs-comment">// and sends the size of each found file on fileSizes.</span>
<span class="hljs-keyword">func</span> walkDir(dir <span class="hljs-typename">string</span>, fileSizes <span class="hljs-keyword">chan</span>&lt;- <span class="hljs-typename">int64</span>) {
<span class="hljs-keyword">for</span> _, entry := <span class="hljs-keyword">range</span> dirents(dir) {
<span class="hljs-keyword">if</span> entry.IsDir() {
subdir := filepath.Join(dir, entry.Name())
walkDir(subdir, fileSizes)
} <span class="hljs-keyword">else</span> {
fileSizes &lt;- entry.Size()
}
}
}
<span class="hljs-comment">// dirents returns the entries of directory dir.</span>
<span class="hljs-keyword">func</span> dirents(dir <span class="hljs-typename">string</span>) []os.FileInfo {
entries, err := ioutil.ReadDir(dir)
<span class="hljs-keyword">if</span> err != <span class="hljs-constant">nil</span> {
fmt.Fprintf(os.Stderr, <span class="hljs-string">&quot;du1: %v\n&quot;</span>, err)
<span class="hljs-keyword">return</span> <span class="hljs-constant">nil</span>
}
<span class="hljs-keyword">return</span> entries
}
</code></pre>
<p>ioutil.ReadDir&#x51FD;&#x6578;&#x6703;&#x8FD4;&#x8FF4;&#x4E00;&#x500B;os.FileInfo&#x985E;&#x578B;&#x7684;slice&#xFF0C;os.FileInfo&#x985E;&#x578B;&#x4E5F;&#x662F;os.Stat&#x9019;&#x500B;&#x51FD;&#x6578;&#x7684;&#x8FD4;&#x8FF4;&#x503C;&#x3002;&#x5C0D;&#x6BCF;&#x4E00;&#x500B;&#x5B50;&#x76EE;&#x5F54;&#x800C;&#x8A00;&#xFF0C;walkDir&#x6703;&#x905E;&#x6B78;&#x5730;&#x8ABF;&#x7528;&#x5176;&#x81EA;&#x8EAB;&#xFF0C;&#x4F75;&#x4E14;&#x6703;&#x5C0D;&#x6BCF;&#x4E00;&#x500B;&#x6587;&#x4EF6;&#x4E5F;&#x905E;&#x6B78;&#x8ABF;&#x7528;&#x3002;walkDir&#x51FD;&#x6578;&#x6703;&#x66CF;fileSizes&#x9019;&#x500B;channel&#x767C;&#x9001;&#x4E00;&#x689D;&#x6D88;&#x606F;&#x3002;&#x9019;&#x689D;&#x6D88;&#x606F;&#x5305;&#x542B;&#x4E86;&#x6587;&#x4EF6;&#x7684;&#x5B57;&#x7BC0;&#x5927;&#x5C0F;&#x3002;</p>
<p>&#x4E0B;&#x9EAB;&#x7684;&#x4E3B;&#x51FD;&#x6578;&#xFF0C;&#x7528;&#x4E86;&#x5169;&#x500B;goroutine&#x3002;&#x5F8C;&#x6AAF;&#x7684;goroutine&#x8ABF;&#x7528;walkDir&#x4F86;&#x904D;&#x6B77;&#x547D;&#x4EE4;&#x884C;&#x7D66;&#x9F63;&#x7684;&#x6BCF;&#x4E00;&#x500B;&#x8DEF;&#x5F91;&#x4F75;&#x6700;&#x7D42;&#x95DC;&#x9589;fileSizes&#x9019;&#x500B;channel&#x3002;&#x4E3B;goroutine&#x6703;&#x5C0D;&#x5176;&#x5F9E;channel&#x4E2D;&#x63A5;&#x6536;&#x5230;&#x7684;&#x6587;&#x4EF6;&#x5927;&#x5C0F;&#x9032;&#x884C;&#x7E8D;&#x52A0;&#xFF0C;&#x4F75;&#x8F38;&#x9F63;&#x5176;&#x548C;&#x3002;</p>
<pre><code class="lang-go"><span class="hljs-keyword">package</span> main
<span class="hljs-keyword">import</span> (
<span class="hljs-string">&quot;flag&quot;</span>
<span class="hljs-string">&quot;fmt&quot;</span>
<span class="hljs-string">&quot;io/ioutil&quot;</span>
<span class="hljs-string">&quot;os&quot;</span>
<span class="hljs-string">&quot;path/filepath&quot;</span>
)
<span class="hljs-keyword">func</span> main() {
<span class="hljs-comment">// Determine the initial directories.</span>
flag.Parse()
roots := flag.Args()
<span class="hljs-keyword">if</span> <span class="hljs-built_in">len</span>(roots) == <span class="hljs-number">0</span> {
roots = []<span class="hljs-typename">string</span>{<span class="hljs-string">&quot;.&quot;</span>}
}
<span class="hljs-comment">// Traverse the file tree.</span>
fileSizes := <span class="hljs-built_in">make</span>(<span class="hljs-keyword">chan</span> <span class="hljs-typename">int64</span>)
<span class="hljs-keyword">go</span> <span class="hljs-keyword">func</span>() {
<span class="hljs-keyword">for</span> _, root := <span class="hljs-keyword">range</span> roots {
walkDir(root, fileSizes)
}
<span class="hljs-built_in">close</span>(fileSizes)
}()
<span class="hljs-comment">// Print the results.</span>
<span class="hljs-keyword">var</span> nfiles, nbytes <span class="hljs-typename">int64</span>
<span class="hljs-keyword">for</span> size := <span class="hljs-keyword">range</span> fileSizes {
nfiles++
nbytes += size
}
printDiskUsage(nfiles, nbytes)
}
<span class="hljs-keyword">func</span> printDiskUsage(nfiles, nbytes <span class="hljs-typename">int64</span>) {
fmt.Printf(<span class="hljs-string">&quot;%d files %.1f GB\n&quot;</span>, nfiles, <span class="hljs-typename">float64</span>(nbytes)/<span class="hljs-number">1e9</span>)
}
</code></pre>
<p>&#x9019;&#x500B;&#x7A0B;&#x5E8F;&#x6703;&#x5728;&#x6253;&#x5370;&#x5176;&#x7D50;&#x679C;&#x4E4B;&#x524D;&#x5361;&#x4F4F;&#x5F88;&#x9577;&#x6642;&#x9593;&#x3002;</p>
<pre><code>$ go build gopl.io/ch8/du1
$ ./du1 $HOME /usr /bin /etc
213201 files 62.7 GB
</code></pre><p>&#x5982;&#x679C;&#x5728;&#x904B;&#x884C;&#x7684;&#x6642;&#x5019;&#x80FD;&#x5920;&#x8B93;&#x6211;&#x5011;&#x77E5;&#x9053;&#x8655;&#x7406;&#x9032;&#x5EA6;&#x7684;&#x8A71;&#x60F3;&#x5FC5;&#x66F4;&#x597D;&#x3002;&#x4F46;&#x662F;&#xFF0C;&#x5982;&#x679C;&#x7C21;&#x55AE;&#x5730;&#x628A;printDiskUsage&#x51FD;&#x6578;&#x8ABF;&#x7528;&#x79FB;&#x52D5;&#x5230;&#x5FAA;&#x74B0;&#x88E1;&#x6703;&#x5C0E;&#x7DFB;&#x5176;&#x6253;&#x5370;&#x9F63;&#x6210;&#x767E;&#x4E0A;&#x97C6;&#x7684;&#x8F38;&#x9F63;&#x3002;</p>
<p>&#x4E0B;&#x9EAB;&#x9019;&#x500B;du&#x7684;&#x8B8A;&#x7A2E;&#x6703;&#x9593;&#x6B47;&#x6253;&#x5370;&#x5167;&#x5BB9;&#xFF0C;&#x4E0D;&#x904E;&#x96BB;&#x6709;&#x5728;&#x8ABF;&#x7528;&#x6642;&#x63D0;&#x4F9B;&#x4E86;-v&#x7684;flag&#x7E94;&#x6703;&#x986F;&#x793A;&#x7A0B;&#x5E8F;&#x9032;&#x5EA6;&#x4FE1;&#x606F;&#x3002;&#x5728;roots&#x76EE;&#x5F54;&#x4E0A;&#x5FAA;&#x74B0;&#x7684;&#x5F8C;&#x6AAF;goroutine&#x5728;&#x9019;&#x88E1;&#x4FDD;&#x6301;&#x4E0D;&#x8B8A;&#x3002;&#x4E3B;goroutine&#x73FE;&#x5728;&#x4F7F;&#x7528;&#x4E86;&#x8A08;&#x6642;&#x5668;&#x4F86;&#x6BCF;500ms&#x751F;&#x6210;&#x4E8B;&#x4EF6;&#xFF0C;&#x7136;&#x5F8C;&#x7528;select&#x8A9E;&#x53E5;&#x4F86;&#x7B49;&#x5F85;&#x6587;&#x4EF6;&#x5927;&#x5C0F;&#x7684;&#x6D88;&#x606F;&#x4F86;&#x66F4;&#x65B0;&#x7E3D;&#x5927;&#x5C0F;&#x6578;&#x64DA;&#xFF0C;&#x6216;&#x8005;&#x4E00;&#x500B;&#x8A08;&#x6642;&#x5668;&#x7684;&#x4E8B;&#x4EF6;&#x4F86;&#x6253;&#x5370;&#x5679;&#x524D;&#x7684;&#x7E3D;&#x5927;&#x5C0F;&#x6578;&#x64DA;&#x3002;&#x5982;&#x679C;-v&#x7684;flag&#x5728;&#x904B;&#x884C;&#x6642;&#x6C92;&#x6709;&#x50B3;&#x5165;&#x7684;&#x8A71;&#xFF0C;tick&#x9019;&#x500B;channel&#x6703;&#x4FDD;&#x6301;&#x7232;nil&#xFF0C;&#x9019;&#x6A23;&#x5728;select&#x88E1;&#x7684;case&#x4E5F;&#x5C31;&#x76F8;&#x5679;&#x65BC;&#x88AB;&#x7981;&#x7528;&#x4E86;&#x3002;</p>
<pre><code class="lang-go">gopl.io/ch8/du2
<span class="hljs-keyword">var</span> verbose = flag.Bool(<span class="hljs-string">&quot;v&quot;</span>, <span class="hljs-constant">false</span>, <span class="hljs-string">&quot;show verbose progress messages&quot;</span>)
<span class="hljs-keyword">func</span> main() {
<span class="hljs-comment">// ...start background goroutine...</span>
<span class="hljs-comment">// Print the results periodically.</span>
<span class="hljs-keyword">var</span> tick &lt;-<span class="hljs-keyword">chan</span> time.Time
<span class="hljs-keyword">if</span> *verbose {
tick = time.Tick(<span class="hljs-number">500</span> * time.Millisecond)
}
<span class="hljs-keyword">var</span> nfiles, nbytes <span class="hljs-typename">int64</span>
loop:
<span class="hljs-keyword">for</span> {
<span class="hljs-keyword">select</span> {
<span class="hljs-keyword">case</span> size, ok := &lt;-fileSizes:
<span class="hljs-keyword">if</span> !ok {
<span class="hljs-keyword">break</span> loop <span class="hljs-comment">// fileSizes was closed</span>
}
nfiles++
nbytes += size
<span class="hljs-keyword">case</span> &lt;-tick:
printDiskUsage(nfiles, nbytes)
}
}
printDiskUsage(nfiles, nbytes) <span class="hljs-comment">// final totals</span>
}
</code></pre>
<p>&#x7531;&#x65BC;&#x6211;&#x5011;&#x7684;&#x7A0B;&#x5E8F;&#x4E0D;&#x518D;&#x4F7F;&#x7528;range&#x5FAA;&#x74B0;&#xFF0C;&#x7B2C;&#x4E00;&#x500B;select&#x7684;case&#x5FC5;&#x9808;&#x986F;&#x5F0F;&#x5730;&#x5224;&#x65B7;fileSizes&#x7684;channel&#x662F;&#x4E0D;&#x662F;&#x5DF2;&#x7D93;&#x88AB;&#x95DC;&#x9589;&#x4E86;&#xFF0C;&#x9019;&#x88E1;&#x53EF;&#x4EE5;&#x7528;&#x5230;channel&#x63A5;&#x6536;&#x7684;&#x4E8C;&#x503C;&#x5F62;&#x5F0F;&#x3002;&#x5982;&#x679C;channel&#x5DF2;&#x7D93;&#x88AB;&#x95DC;&#x9589;&#x4E86;&#x7684;&#x8A71;&#xFF0C;&#x7A0B;&#x5E8F;&#x6703;&#x76F4;&#x63A5;&#x9000;&#x9F63;&#x5FAA;&#x74B0;&#x3002;&#x9019;&#x88E1;&#x7684;break&#x8A9E;&#x53E5;&#x7528;&#x5230;&#x4E86;&#x6A19;&#x7C64;break&#xFF0C;&#x9019;&#x6A23;&#x53EF;&#x4EE5;&#x8855;&#x6642;&#x7D42;&#x7D50;select&#x548C;for&#x5169;&#x500B;&#x5FAA;&#x74B0;&#xFF1B;&#x5982;&#x679C;&#x6C92;&#x6709;&#x7528;&#x6A19;&#x7C64;&#x5C31;break&#x7684;&#x8A71;&#x96BB;&#x6703;&#x9000;&#x9F63;&#x5167;&#x5C64;&#x7684;select&#x5FAA;&#x74B0;&#xFF0C;&#x800C;&#x5916;&#x5C64;&#x7684;for&#x5FAA;&#x74B0;&#x6703;&#x4F7F;&#x4E4B;&#x9032;&#x5165;&#x4E0B;&#x4E00;&#x8F2A;select&#x5FAA;&#x74B0;&#x3002;</p>
<p>&#x73FE;&#x5728;&#x7A0B;&#x5E8F;&#x6703;&#x60A0;&#x9592;&#x5730;&#x7232;&#x6211;&#x5011;&#x6253;&#x5370;&#x66F4;&#x65B0;&#x6D41;&#xFF1A;</p>
<pre><code>$ go build gopl.io/ch8/du2
$ ./du2 -v $HOME /usr /bin /etc
28608 files 8.3 GB
54147 files 10.3 GB
93591 files 15.1 GB
127169 files 52.9 GB
175931 files 62.2 GB
213201 files 62.7 GB
</code></pre><p>&#x7136;&#x800C;&#x9019;&#x500B;&#x7A0B;&#x5E8F;&#x9084;&#x662F;&#x6703;&#x82B1;&#x4E0A;&#x5F88;&#x9577;&#x6642;&#x9593;&#x7E94;&#x6703;&#x7D50;&#x675F;&#x3002;&#x7121;&#x6CD5;&#x5C0D;walkDir&#x505A;&#x4F75;&#x884C;&#x5316;&#x8655;&#x7406;&#x6C92;&#x4EC0;&#x9EBD;&#x5F46;&#x7684;&#x539F;&#x56E0;&#xFF0C;&#x7121;&#x975E;&#x662F;&#x56E0;&#x7232;&#x78C1;&#x76E4;&#x7E6B;&#x7D71;&#x4F75;&#x884C;&#x9650;&#x88FD;&#x3002;&#x4E0B;&#x9EAB;&#x9019;&#x500B;&#x7B2C;&#x4E09;&#x500B;&#x7248;&#x672C;&#x7684;du&#xFF0C;&#x6703;&#x5C0D;&#x6BCF;&#x4E00;&#x500B;walkDir&#x7684;&#x8ABF;&#x7528;&#x5275;&#x5EFA;&#x4E00;&#x500B;&#x65B0;&#x7684;goroutine&#x3002;&#x5B83;&#x4F7F;&#x7528;sync.WaitGroup (&#xA7;8.5)&#x4F86;&#x5C0D;&#x4ECD;&#x820A;&#x6D3B;&#x8E8D;&#x7684;walkDir&#x8ABF;&#x7528;&#x9032;&#x884C;&#x8A08;&#x6578;&#xFF0C;&#x53E6;&#x4E00;&#x500B;goroutine&#x6703;&#x5728;&#x8A08;&#x6578;&#x5668;&#x6E1B;&#x7232;&#x96F6;&#x7684;&#x6642;&#x5019;&#x5C07;fileSizes&#x9019;&#x500B;channel&#x95DC;&#x9589;&#x3002;</p>
<pre><code class="lang-go">gopl.io/ch8/du3
<span class="hljs-keyword">func</span> main() {
<span class="hljs-comment">// ...determine roots...</span>
<span class="hljs-comment">// Traverse each root of the file tree in parallel.</span>
fileSizes := <span class="hljs-built_in">make</span>(<span class="hljs-keyword">chan</span> <span class="hljs-typename">int64</span>)
<span class="hljs-keyword">var</span> n sync.WaitGroup
<span class="hljs-keyword">for</span> _, root := <span class="hljs-keyword">range</span> roots {
n.Add(<span class="hljs-number">1</span>)
<span class="hljs-keyword">go</span> walkDir(root, &amp;n, fileSizes)
}
<span class="hljs-keyword">go</span> <span class="hljs-keyword">func</span>() {
n.Wait()
<span class="hljs-built_in">close</span>(fileSizes)
}()
<span class="hljs-comment">// ...select loop...</span>
}
<span class="hljs-keyword">func</span> walkDir(dir <span class="hljs-typename">string</span>, n *sync.WaitGroup, fileSizes <span class="hljs-keyword">chan</span>&lt;- <span class="hljs-typename">int64</span>) {
<span class="hljs-keyword">defer</span> n.Done()
<span class="hljs-keyword">for</span> _, entry := <span class="hljs-keyword">range</span> dirents(dir) {
<span class="hljs-keyword">if</span> entry.IsDir() {
n.Add(<span class="hljs-number">1</span>)
subdir := filepath.Join(dir, entry.Name())
<span class="hljs-keyword">go</span> walkDir(subdir, n, fileSizes)
} <span class="hljs-keyword">else</span> {
fileSizes &lt;- entry.Size()
}
}
}
</code></pre>
<p>&#x7531;&#x65BC;&#x9019;&#x500B;&#x7A0B;&#x5E8F;&#x5728;&#x9AD8;&#x5CEF;&#x671F;&#x6703;&#x5275;&#x5EFA;&#x6210;&#x767E;&#x4E0A;&#x97C6;&#x7684;goroutine&#xFF0C;&#x6211;&#x5011;&#x9700;&#x8981;&#x8129;&#x6539;dirents&#x51FD;&#x6578;&#xFF0C;&#x7528;&#x8A08;&#x6578;&#x4FE1;&#x865F;&#x91CF;&#x4F86;&#x963B;&#x6B62;&#x4ED6;&#x8855;&#x6642;&#x6253;&#x958B;&#x592A;&#x591A;&#x7684;&#x6587;&#x4EF6;&#xFF0C;&#x5C31;&#x50CF;&#x6211;&#x5011;&#x5728;8.7&#x7BC0;&#x4E2D;&#x7684;&#x4F75;&#x767C;&#x722C;&#x87F2;&#x4E00;&#x6A23;&#xFF1A;</p>
<pre><code class="lang-go"><span class="hljs-comment">// sema is a counting semaphore for limiting concurrency in dirents.</span>
<span class="hljs-keyword">var</span> sema = <span class="hljs-built_in">make</span>(<span class="hljs-keyword">chan</span> <span class="hljs-keyword">struct</span>{}, <span class="hljs-number">20</span>)
<span class="hljs-comment">// dirents returns the entries of directory dir.</span>
<span class="hljs-keyword">func</span> dirents(dir <span class="hljs-typename">string</span>) []os.FileInfo {
sema &lt;- <span class="hljs-keyword">struct</span>{}{} <span class="hljs-comment">// acquire token</span>
<span class="hljs-keyword">defer</span> <span class="hljs-keyword">func</span>() { &lt;-sema }() <span class="hljs-comment">// release token</span>
<span class="hljs-comment">// ...</span>
</code></pre>
<p>&#x9019;&#x500B;&#x7248;&#x672C;&#x6BD4;&#x4E4B;&#x524D;&#x90A3;&#x500B;&#x5FEB;&#x4E86;&#x597D;&#x5E7E;&#x500D;&#xFF0C;&#x5118;&#x7BA1;&#x5176;&#x5177;&#x9AD4;&#x6548;&#x7387;&#x9084;&#x662F;&#x548C;&#x4F60;&#x7684;&#x904B;&#x884C;&#x74B0;&#x5883;&#xFF0C;&#x6A5F;&#x5668;&#x914D;&#x7F6E;&#x76F8;&#x95DC;&#x3002;</p>
<p>&#x7DF4;&#x7FD2;8.9: &#x7DE8;&#x5BEB;&#x4E00;&#x500B;du&#x5DE5;&#x5177;&#xFF0C;&#x6BCF;&#x9694;&#x4E00;&#x6BB5;&#x6642;&#x9593;&#x5C07;root&#x76EE;&#x5F54;&#x4E0B;&#x7684;&#x76EE;&#x5F54;&#x5927;&#x5C0F;&#x8A08;&#x7B97;&#x4F75;&#x986F;&#x793A;&#x9F63;&#x4F86;&#x3002;</p>
</section>

View File

@@ -48,7 +48,7 @@
<body>
<div class="book" data-level="8.9" data-chapter-title="併髮的退齣" data-filepath="ch8/ch8-09.md" data-basepath=".." data-revision="Wed Dec 09 2015 15:54:13 GMT+0800 (中国标准时间)">
<div class="book" data-level="8.9" data-chapter-title="併髮的退齣" data-filepath="ch8/ch8-09.md" data-basepath=".." data-revision="Mon Dec 14 2015 11:30:54 GMT+0800 (中国标准时间)">
<div class="book-summary">
@@ -2060,7 +2060,71 @@
<section class="normal" id="section-">
<h2 id="89-&#x4F75;&#x9AEE;&#x7684;&#x9000;&#x9F63;">8.9. &#x4F75;&#x9AEE;&#x7684;&#x9000;&#x9F63;</h2>
<p>TODO</p>
<p>&#x6709;&#x6642;&#x5019;&#x6211;&#x5011;&#x9700;&#x8981;&#x901A;&#x77E5;goroutine&#x505C;&#x6B62;&#x5B83;&#x6B63;&#x5728;&#x4E7E;&#x7684;&#x4E8B;&#x60C5;&#xFF0C;&#x6BD4;&#x5982;&#x4E00;&#x7B87;&#x6B63;&#x5728;&#x57F7;&#x884C;&#x8A08;&#x7B97;&#x7684;web&#x670D;&#x52D9;&#xFF0C;&#x7136;&#x800C;&#x5B83;&#x7684;&#x5BA2;&#x6236;&#x7AEF;&#x5DF2;&#x7D93;&#x65B7;&#x958B;&#x4E86;&#x548C;&#x670D;&#x52D9;&#x7AEF;&#x7684;&#x9023;&#x63A5;&#x3002;</p>
<p>Go&#x8A9E;&#x8A00;&#x4F75;&#x6C92;&#x6709;&#x63D0;&#x4F9B;&#x5728;&#x4E00;&#x7B87;goroutine&#x4E2D;&#x7D42;&#x6B62;&#x53E6;&#x4E00;&#x7B87;goroutine&#x7684;&#x65B9;&#x6CD5;&#xFF0C;&#x7531;&#x65BC;&#x9019;&#x6A23;&#x6703;&#x5C0E;&#x7DFB;goroutine&#x4E4B;&#x9593;&#x7684;&#x5171;&#x4EAB;&#x8B8A;&#x91CF;&#x843D;&#x5728;&#x672A;&#x5B9A;&#x7FA9;&#x7684;&#x72C0;&#x614B;&#x4E0A;&#x3002;&#x5728;8.7&#x7BC0;&#x4E2D;&#x7684;rocket launch&#x7A0B;&#x5E8F;&#x4E2D;&#xFF0C;&#x6211;&#x5011;&#x5F80;&#x540D;&#x5B57;&#x53EB;abort&#x7684;channel&#x88E1;&#x767C;&#x9001;&#x4E86;&#x4E00;&#x7B87;&#x7C21;&#x55AE;&#x7684;&#x503C;&#xFF0C;&#x5728;countdown&#x7684;goroutine&#x4E2D;&#x6703;&#x628A;&#x9019;&#x7B87;&#x503C;&#x7406;&#x89E3;&#x7232;&#x81EA;&#x5DF1;&#x7684;&#x9000;&#x9F63;&#x4FE1;&#x865F;&#x3002;&#x4F46;&#x662F;&#x5982;&#x679C;&#x6211;&#x5011;&#x60F3;&#x8981;&#x9000;&#x9F63;&#x5169;&#x7B87;&#x6216;&#x8005;&#x4EFB;&#x610F;&#x591A;&#x7B87;goroutine&#x600E;&#x9EBC;&#x8FA6;&#x5462;&#xFF1F;</p>
<p>&#x4E00;&#x7A2E;&#x53EF;&#x80FD;&#x7684;&#x624B;&#x6BB5;&#x662F;&#x56AE;abort&#x7684;channel&#x88E1;&#x767C;&#x9001;&#x548C;goroutine&#x6578;&#x76EE;&#x4E00;&#x6A23;&#x591A;&#x7684;&#x4E8B;&#x4EF6;&#x4F86;&#x9000;&#x9F63;&#x5B83;&#x5011;&#x3002;&#x5982;&#x679C;&#x9019;&#x4E9B;goroutine&#x4E2D;&#x5DF2;&#x7D93;&#x6709;&#x4E00;&#x4E9B;&#x81EA;&#x5DF1;&#x9000;&#x9F63;&#x4E86;&#xFF0C;&#x90A3;&#x9EBC;&#x6703;&#x5C0E;&#x7DFB;&#x6211;&#x5011;&#x7684;channel&#x88E1;&#x7684;&#x4E8B;&#x4EF6;&#x6578;&#x6BD4;goroutine&#x9084;&#x591A;&#xFF0C;&#x9019;&#x6A23;&#x5C0E;&#x7DFB;&#x6211;&#x5011;&#x7684;&#x767C;&#x9001;&#x76F4;&#x63A5;&#x88AB;&#x963B;&#x585E;&#x3002;&#x53E6;&#x4E00;&#x65B9;&#x9EAB;&#xFF0C;&#x5982;&#x679C;&#x9019;&#x4E9B;goroutine&#x53C8;&#x751F;&#x6210;&#x4E86;&#x5176;&#x5B83;&#x7684;goroutine&#xFF0C;&#x6211;&#x5011;&#x7684;channel&#x88E1;&#x7684;&#x6578;&#x76EE;&#x53C8;&#x592A;&#x5C11;&#x4E86;&#xFF0C;&#x6240;&#x4EE5;&#x6709;&#x4E9B;goroutine&#x53EF;&#x80FD;&#x6703;&#x7121;&#x6CD5;&#x63A5;&#x6536;&#x5230;&#x9000;&#x9F63;&#x6D88;&#x606F;&#x3002;&#x4E00;&#x822C;&#x60C5;&#x6CC1;&#x4E0B;&#x6211;&#x5011;&#x662F;&#x5F88;&#x96E3;&#x77E5;&#x9053;&#x5728;&#x67D0;&#x4E00;&#x7B87;&#x6642;&#x523B;&#x5177;&#x9AD4;&#x6709;&#x591A;&#x5C11;&#x7B87;goroutine&#x5728;&#x904B;&#x884C;&#x7740;&#x7684;&#x3002;&#x53E6;&#x5916;&#xFF0C;&#x5679;&#x4E00;&#x7B87;goroutine&#x5F9E;abort channel&#x4E2D;&#x63A5;&#x6536;&#x5230;&#x4E00;&#x7B87;&#x503C;&#x7684;&#x6642;&#x5019;&#xFF0C;&#x4ED6;&#x6703;&#x6D88;&#x8CBB;&#x6389;&#x9019;&#x7B87;&#x503C;&#xFF0C;&#x9019;&#x6A23;&#x5176;&#x5B83;&#x7684;goroutine&#x5C31;&#x6C92;&#x6CD5;&#x770B;&#x5230;&#x9019;&#x689D;&#x4FE1;&#x606F;&#x3002;&#x7232;&#x4E86;&#x80FD;&#x5920;&#x9054;&#x5230;&#x6211;&#x5011;&#x9000;&#x9F63;goroutine&#x7684;&#x76EE;&#x7684;&#xFF0C;&#x6211;&#x5011;&#x9700;&#x8981;&#x66F4;&#x9760;&#x8B5C;&#x7684;&#x7B56;&#x7565;&#xFF0C;&#x4F86;&#x901A;&#x904E;&#x4E00;&#x7B87;channel&#x628A;&#x6D88;&#x606F;&#x5EE3;&#x64AD;&#x9F63;&#x53BB;&#xFF0C;&#x9019;&#x6A23;goroutine&#x5011;&#x80FD;&#x5920;&#x770B;&#x5230;&#x9019;&#x689D;&#x4E8B;&#x4EF6;&#x6D88;&#x606F;&#xFF0C;&#x4F75;&#x4E14;&#x5728;&#x4E8B;&#x4EF6;&#x5B8C;&#x6210;&#x4E4B;&#x5F8C;&#xFF0C;&#x53EF;&#x4EE5;&#x77E5;&#x9053;&#x9019;&#x4EF6;&#x4E8B;&#x5DF2;&#x7D93;&#x767C;&#x751F;&#x904E;&#x4E86;&#x3002;</p>
<p>&#x8FF4;&#x61B6;&#x4E00;&#x4E0B;&#x6211;&#x5011;&#x95DC;&#x9589;&#x4E86;&#x4E00;&#x7B87;channel&#x4F75;&#x4E14;&#x88AB;&#x6D88;&#x8CBB;&#x6389;&#x4E86;&#x6240;&#x6709;&#x5DF2;&#x767C;&#x9001;&#x7684;&#x503C;&#xFF0C;&#x64CD;&#x4F5C;channel&#x4E4B;&#x5F8C;&#x7684;&#x4EE3;&#x78BC;&#x53EF;&#x4EE5;&#x7ACB;&#x537D;&#x88AB;&#x57F7;&#x884C;&#xFF0C;&#x4F75;&#x4E14;&#x6703;&#x7522;&#x751F;&#x96F6;&#x503C;&#x3002;&#x6211;&#x5011;&#x53EF;&#x4EE5;&#x5C07;&#x9019;&#x7B87;&#x6A5F;&#x88FD;&#x64F4;&#x5C55;&#x4E00;&#x4E0B;&#xFF0C;&#x4F86;&#x4F5C;&#x7232;&#x6211;&#x5011;&#x7684;&#x5EE3;&#x64AD;&#x6A5F;&#x88FD;&#xFF1A;&#x4E0D;&#x8981;&#x56AE;channel&#x767C;&#x9001;&#x503C;&#xFF0C;&#x800C;&#x662F;&#x7528;&#x95DC;&#x9589;&#x4E00;&#x7B87;channel&#x4F86;&#x9032;&#x884C;&#x5EE3;&#x64AD;&#x3002;</p>
<p>&#x96BB;&#x8981;&#x4E00;&#x4E9B;&#x5C0F;&#x8129;&#x6539;&#xFF0C;&#x6211;&#x5011;&#x5C31;&#x53EF;&#x4EE5;&#x628A;&#x9000;&#x9F63;&#x908F;&#x8F2F;&#x52A0;&#x5165;&#x5230;&#x524D;&#x4E00;&#x7BC0;&#x7684;du&#x7A0B;&#x5E8F;&#x3002;&#x9996;&#x5148;&#xFF0C;&#x6211;&#x5011;&#x5275;&#x5EFA;&#x4E00;&#x7B87;&#x9000;&#x9F63;&#x7684;channel&#xFF0C;&#x9019;&#x7B87;channel&#x4E0D;&#x6703;&#x56AE;&#x5176;&#x4E2D;&#x767C;&#x9001;&#x4EFB;&#x4F55;&#x503C;&#xFF0C;&#x4F46;&#x5176;&#x6240;&#x5728;&#x7684;&#x9589;&#x5305;&#x5167;&#x8981;&#x5BEB;&#x660E;&#x7A0B;&#x5E8F;&#x9700;&#x8981;&#x9000;&#x9F63;&#x3002;&#x6211;&#x5011;&#x8855;&#x6642;&#x9084;&#x5B9A;&#x7FA9;&#x4E86;&#x4E00;&#x7B87;&#x5DE5;&#x5177;&#x51FD;&#x6578;&#xFF0C;cancelled&#xFF0C;&#x9019;&#x7B87;&#x51FD;&#x6578;&#x5728;&#x88AB;&#x8ABF;&#x7528;&#x7684;&#x6642;&#x5019;&#x6703;&#x8F2A;&#x8A62;&#x9000;&#x9F63;&#x72C0;&#x614B;&#x3002;</p>
<pre><code class="lang-go">gopl.io/ch8/du4
<span class="hljs-keyword">var</span> done = <span class="hljs-built_in">make</span>(<span class="hljs-keyword">chan</span> <span class="hljs-keyword">struct</span>{})
<span class="hljs-keyword">func</span> cancelled() <span class="hljs-typename">bool</span> {
<span class="hljs-keyword">select</span> {
<span class="hljs-keyword">case</span> &lt;-done:
<span class="hljs-keyword">return</span> <span class="hljs-constant">true</span>
<span class="hljs-keyword">default</span>:
<span class="hljs-keyword">return</span> <span class="hljs-constant">false</span>
}
}
</code></pre>
<p>&#x4E0B;&#x9EAB;&#x6211;&#x5011;&#x5275;&#x5EFA;&#x4E00;&#x7B87;&#x5F9E;&#x6A19;&#x6E96;&#x8F38;&#x5165;&#x6D41;&#x4E2D;&#x8B80;&#x53D6;&#x5167;&#x5BB9;&#x7684;goroutine&#xFF0C;&#x9019;&#x662F;&#x4E00;&#x7B87;&#x6BD4;&#x8F03;&#x5178;&#x578B;&#x7684;&#x9023;&#x63A5;&#x5230;&#x7D42;&#x7AEF;&#x7684;&#x7A0B;&#x5E8F;&#x3002;&#x6BCF;&#x5679;&#x6709;&#x8F38;&#x5165;&#x88AB;&#x8B80;&#x5230;(&#x6BD4;&#x5982;&#x7528;&#x6236;&#x6309;&#x4E86;&#x8FF4;&#x8ECA;&#x9375;)&#xFF0C;&#x9019;&#x7B87;goroutine&#x5C31;&#x6703;&#x628A;&#x53D6;&#x6D88;&#x6D88;&#x606F;&#x901A;&#x904E;&#x95DC;&#x9589;done&#x7684;channel&#x5EE3;&#x64AD;&#x9F63;&#x53BB;&#x3002;</p>
<pre><code class="lang-go"><span class="hljs-comment">// Cancel traversal when input is detected.</span>
<span class="hljs-keyword">go</span> <span class="hljs-keyword">func</span>() {
os.Stdin.Read(<span class="hljs-built_in">make</span>([]<span class="hljs-typename">byte</span>, <span class="hljs-number">1</span>)) <span class="hljs-comment">// read a single byte</span>
<span class="hljs-built_in">close</span>(done)
}()
</code></pre>
<p>&#x73FE;&#x5728;&#x6211;&#x5011;&#x9700;&#x8981;&#x4F7F;&#x6211;&#x5011;&#x7684;goroutine&#x4F86;&#x5C0D;&#x53D6;&#x6D88;&#x9032;&#x884C;&#x97FF;&#x61C9;&#x3002;&#x5728;main goroutine&#x4E2D;&#xFF0C;&#x6211;&#x5011;&#x6DFB;&#x52A0;&#x4E86;select&#x7684;&#x7B2C;&#x4E09;&#x7B87;case&#x8A9E;&#x53E5;&#xFF0C;&#x5617;&#x8A66;&#x5F9E;done channel&#x4E2D;&#x63A5;&#x6536;&#x5167;&#x5BB9;&#x3002;&#x5982;&#x679C;&#x9019;&#x7B87;case&#x88AB;&#x6EFF;&#x8DB3;&#x7684;&#x8A71;&#xFF0C;&#x5728;select&#x5230;&#x7684;&#x6642;&#x5019;&#x537D;&#x6703;&#x8FD4;&#x8FF4;&#xFF0C;&#x4F46;&#x5728;&#x7D50;&#x675F;&#x4E4B;&#x524D;&#x6211;&#x5011;&#x9700;&#x8981;&#x628A;fileSizes channel&#x4E2D;&#x7684;&#x5167;&#x5BB9;&#x201C;&#x6392;&#x201D;&#x7A7A;&#xFF0C;&#x5728;channel&#x88AB;&#x95DC;&#x9589;&#x4E4B;&#x524D;&#xFF0C;&#x6368;&#x68C4;&#x6389;&#x6240;&#x6709;&#x503C;&#x3002;&#x9019;&#x6A23;&#x53EF;&#x4EE5;&#x4FDD;&#x8A3C;&#x5C0D;walkDir&#x7684;&#x8ABF;&#x7528;&#x4E0D;&#x8981;&#x88AB;&#x56AE;fileSizes&#x767C;&#x9001;&#x4FE1;&#x606F;&#x963B;&#x585E;&#x4F4F;&#xFF0C;&#x53EF;&#x4EE5;&#x6B63;&#x78BA;&#x5730;&#x5B8C;&#x6210;&#x3002;</p>
<pre><code class="lang-go"><span class="hljs-keyword">for</span> {
<span class="hljs-keyword">select</span> {
<span class="hljs-keyword">case</span> &lt;-done:
<span class="hljs-comment">// Drain fileSizes to allow existing goroutines to finish.</span>
<span class="hljs-keyword">for</span> <span class="hljs-keyword">range</span> fileSizes {
<span class="hljs-comment">// Do nothing.</span>
}
<span class="hljs-keyword">return</span>
<span class="hljs-keyword">case</span> size, ok := &lt;-fileSizes:
<span class="hljs-comment">// ...</span>
}
}
</code></pre>
<p>walkDir&#x9019;&#x7B87;goroutine&#x4E00;&#x555F;&#x52D5;&#x5C31;&#x6703;&#x8F2A;&#x8A62;&#x53D6;&#x6D88;&#x72C0;&#x614B;&#xFF0C;&#x5982;&#x679C;&#x53D6;&#x6D88;&#x72C0;&#x614B;&#x88AB;&#x8A2D;&#x7F6E;&#x7684;&#x8A71;&#x6703;&#x76F4;&#x63A5;&#x8FD4;&#x8FF4;&#xFF0C;&#x4F75;&#x4E14;&#x4E0D;&#x505A;&#x984D;&#x5916;&#x7684;&#x4E8B;&#x60C5;&#x3002;&#x9019;&#x6A23;&#x6211;&#x5011;&#x5C07;&#x6240;&#x6709;&#x5728;&#x53D6;&#x6D88;&#x4E8B;&#x4EF6;&#x4E4B;&#x5F8C;&#x5275;&#x5EFA;&#x7684;goroutine&#x6539;&#x8B8A;&#x7232;&#x7121;&#x64CD;&#x4F5C;&#x3002;</p>
<pre><code class="lang-go"><span class="hljs-keyword">func</span> walkDir(dir <span class="hljs-typename">string</span>, n *sync.WaitGroup, fileSizes <span class="hljs-keyword">chan</span>&lt;- <span class="hljs-typename">int64</span>) {
<span class="hljs-keyword">defer</span> n.Done()
<span class="hljs-keyword">if</span> cancelled() {
<span class="hljs-keyword">return</span>
}
<span class="hljs-keyword">for</span> _, entry := <span class="hljs-keyword">range</span> dirents(dir) {
<span class="hljs-comment">// ...</span>
}
}
</code></pre>
<p>&#x5728;walkDir&#x51FD;&#x6578;&#x7684;&#x5FAA;&#x74B0;&#x4E2D;&#x6211;&#x5011;&#x5C0D;&#x53D6;&#x6D88;&#x72C0;&#x614B;&#x9032;&#x884C;&#x8F2A;&#x8A62;&#x53EF;&#x4EE5;&#x5E36;&#x4F86;&#x660E;&#x986F;&#x7684;&#x76CA;&#x8655;&#xFF0C;&#x53EF;&#x4EE5;&#x907F;&#x514D;&#x5728;&#x53D6;&#x6D88;&#x4E8B;&#x4EF6;&#x767C;&#x751F;&#x6642;&#x9084;&#x53BB;&#x5275;&#x5EFA;goroutine&#x3002;&#x53D6;&#x6D88;&#x672C;&#x8EAB;&#x662F;&#x6709;&#x4E00;&#x4E9B;&#x4EE3;&#x50F9;&#x7684;&#xFF1B;&#x60F3;&#x8981;&#x5FEB;&#x901F;&#x7684;&#x97FF;&#x61C9;&#x9700;&#x8981;&#x5C0D;&#x7A0B;&#x5E8F;&#x908F;&#x8F2F;&#x9032;&#x884C;&#x4FB5;&#x5165;&#x5F0F;&#x7684;&#x8129;&#x6539;&#x3002;&#x78BA;&#x4FDD;&#x5728;&#x53D6;&#x6D88;&#x767C;&#x751F;&#x4E4B;&#x5F8C;&#x4E0D;&#x8981;&#x6709;&#x4EE3;&#x50F9;&#x592A;&#x5927;&#x7684;&#x64CD;&#x4F5C;&#x53EF;&#x80FD;&#x6703;&#x9700;&#x8981;&#x8129;&#x6539;&#x4F60;&#x4EE3;&#x78BC;&#x88E1;&#x7684;&#x5F88;&#x591A;&#x5730;&#x65B9;&#xFF0C;&#x4F46;&#x662F;&#x5728;&#x4E00;&#x4E9B;&#x91CD;&#x8981;&#x7684;&#x5730;&#x65B9;&#x53BB;&#x6AA2;&#x67FB;&#x53D6;&#x6D88;&#x4E8B;&#x4EF6;&#x4E5F;&#x78BA;&#x5BE6;&#x80FD;&#x5E36;&#x4F86;&#x5F88;&#x5927;&#x7684;&#x597D;&#x8655;&#x3002;</p>
<p>&#x5C0D;&#x9019;&#x7B87;&#x7A0B;&#x5E8F;&#x7684;&#x4E00;&#x7B87;&#x7C21;&#x55AE;&#x7684;&#x6027;&#x80FD;&#x5206;&#x6790;&#x53EF;&#x4EE5;&#x63ED;&#x793A;&#x74F6;&#x9838;&#x5728;dirents&#x51FD;&#x6578;&#x4E2D;&#x7A6B;&#x53D6;&#x4E00;&#x7B87;&#x4FE1;&#x865F;&#x91CF;&#x3002;&#x4E0B;&#x9EAB;&#x7684;select&#x53EF;&#x4EE5;&#x8B93;&#x9019;&#x7A2E;&#x64CD;&#x4F5C;&#x53EF;&#x4EE5;&#x88AB;&#x53D6;&#x6D88;&#xFF0C;&#x4F75;&#x4E14;&#x53EF;&#x4EE5;&#x5C07;&#x53D6;&#x6D88;&#x6642;&#x7684;&#x5EF6;&#x9072;&#x5F9E;&#x5E7E;&#x767E;&#x6BEB;&#x79D2;&#x964D;&#x4F4E;&#x5230;&#x5E7E;&#x5341;&#x6BEB;&#x79D2;&#x3002;</p>
<pre><code class="lang-go"><span class="hljs-keyword">func</span> dirents(dir <span class="hljs-typename">string</span>) []os.FileInfo {
<span class="hljs-keyword">select</span> {
<span class="hljs-keyword">case</span> sema &lt;- <span class="hljs-keyword">struct</span>{}{}: <span class="hljs-comment">// acquire token</span>
<span class="hljs-keyword">case</span> &lt;-done:
<span class="hljs-keyword">return</span> <span class="hljs-constant">nil</span> <span class="hljs-comment">// cancelled</span>
}
<span class="hljs-keyword">defer</span> <span class="hljs-keyword">func</span>() { &lt;-sema }() <span class="hljs-comment">// release token</span>
<span class="hljs-comment">// ...read directory...</span>
}
</code></pre>
<p>&#x73FE;&#x5728;&#x5679;&#x53D6;&#x6D88;&#x767C;&#x751F;&#x6642;&#xFF0C;&#x6240;&#x6709;&#x5F8C;&#x6AAF;&#x7684;goroutine&#x90FD;&#x6703;&#x8FC5;&#x901F;&#x505C;&#x6B62;&#x4F75;&#x4E14;&#x4E3B;&#x51FD;&#x6578;&#x6703;&#x8FD4;&#x8FF4;&#x3002;&#x5679;&#x7136;&#xFF0C;&#x5679;&#x4E3B;&#x51FD;&#x6578;&#x8FD4;&#x8FF4;&#x6642;&#xFF0C;&#x4E00;&#x7B87;&#x7A0B;&#x5E8F;&#x6703;&#x9000;&#x9F63;&#xFF0C;&#x800C;&#x6211;&#x5011;&#x53C8;&#x7121;&#x6CD5;&#x5728;&#x4E3B;&#x51FD;&#x6578;&#x9000;&#x9F63;&#x7684;&#x6642;&#x5019;&#x78BA;&#x8A8D;&#x5176;&#x5DF2;&#x7D93;&#x91CB;&#x653E;&#x4E86;&#x6240;&#x6709;&#x7684;&#x8CC7;&#x6E90;(&#x8B6F;&#x8A3B;&#xFF1A;&#x56E0;&#x7232;&#x7A0B;&#x5E8F;&#x90FD;&#x9000;&#x9F63;&#x4E86;&#xFF0C;&#x4F60;&#x7684;&#x4EE3;&#x78BC;&#x90FD;&#x6C92;&#x6CD5;&#x57F7;&#x884C;&#x4E86;)&#x3002;&#x9019;&#x88E1;&#x6709;&#x4E00;&#x7B87;&#x65B9;&#x4FBF;&#x7684;&#x7AC5;&#x9580;&#x6211;&#x5011;&#x53EF;&#x4EE5;&#x4E00;&#x7528;&#xFF1A;&#x53D6;&#x4EE3;&#x6389;&#x76F4;&#x63A5;&#x5F9E;&#x4E3B;&#x51FD;&#x6578;&#x8FD4;&#x8FF4;&#xFF0C;&#x6211;&#x5011;&#x8ABF;&#x7528;&#x4E00;&#x7B87;panic&#xFF0C;&#x7136;&#x5F8C;runtime&#x6703;&#x628A;&#x6BCF;&#x4E00;&#x7B87;goroutine&#x7684;&#x68E7;dump&#x4E0B;&#x4F86;&#x3002;&#x5982;&#x679C;main goroutine&#x662F;&#x552F;&#x4E00;&#x4E00;&#x7B87;&#x5269;&#x4E0B;&#x7684;goroutine&#x7684;&#x8A71;&#xFF0C;&#x4ED6;&#x6703;&#x6E05;&#x7406;&#x6389;&#x81EA;&#x5DF1;&#x7684;&#x4E00;&#x5207;&#x8CC7;&#x6E90;&#x3002;&#x4F46;&#x662F;&#x5982;&#x679C;&#x9084;&#x6709;&#x5176;&#x5B83;&#x7684;goroutine&#x6C92;&#x6709;&#x9000;&#x9F63;&#xFF0C;&#x4ED6;&#x5011;&#x53EF;&#x80FD;&#x6C92;&#x8FA6;&#x6CD5;&#x88AB;&#x6B63;&#x78BA;&#x5730;&#x53D6;&#x6D88;&#x6389;&#xFF0C;&#x4E5F;&#x6709;&#x53EF;&#x80FD;&#x88AB;&#x53D6;&#x6D88;&#x4F46;&#x662F;&#x53D6;&#x6D88;&#x64CD;&#x4F5C;&#x6703;&#x5F88;&#x82B1;&#x6642;&#x9593;&#xFF1B;&#x6240;&#x4EE5;&#x9019;&#x88E1;&#x7684;&#x4E00;&#x7B87;&#x8ABF;&#x7814;&#x9084;&#x662F;&#x5F88;&#x6709;&#x5FC5;&#x8981;&#x7684;&#x3002;&#x6211;&#x5011;&#x7528;panic&#x4F86;&#x7A6B;&#x53D6;&#x5230;&#x8DB3;&#x5920;&#x7684;&#x4FE1;&#x606F;&#x4F86;&#x9A57;&#x8A3C;&#x6211;&#x5011;&#x4E0A;&#x9EAB;&#x7684;&#x5224;&#x65B7;&#xFF0C;&#x770B;&#x770B;&#x6700;&#x7D42;&#x5230;&#x5E95;&#x662F;&#x4EC0;&#x9EBC;&#x6A23;&#x7684;&#x60C5;&#x6CC1;&#x3002;</p>
<p>&#x7DF4;&#x7FD2;8.10: HTTP&#x8ACB;&#x6C42;&#x53EF;&#x80FD;&#x6703;&#x56E0;http.Request&#x7D50;&#x69CB;&#x9AD4;&#x4E2D;Cancel channel&#x7684;&#x95DC;&#x9589;&#x800C;&#x53D6;&#x6D88;&#x3002;&#x8129;&#x6539;8.6&#x7BC0;&#x4E2D;&#x7684;web crawler&#x4F86;&#x652F;&#x6301;&#x53D6;&#x6D88;http&#x8ACB;&#x6C42;&#x3002;</p>
<p>&#x63D0;&#x793A;: http.Get&#x4F75;&#x6C92;&#x6709;&#x63D0;&#x4F9B;&#x65B9;&#x4FBF;&#x5730;&#x5B9A;&#x88FD;&#x4E00;&#x7B87;&#x8ACB;&#x6C42;&#x7684;&#x65B9;&#x6CD5;&#x3002;&#x4F60;&#x53EF;&#x4EE5;&#x7528;http.NewRequest&#x4F86;&#x53D6;&#x800C;&#x4EE3;&#x4E4B;&#xFF0C;&#x8A2D;&#x7F6E;&#x5B83;&#x7684;Cancel&#x5B57;&#x6BB5;&#xFF0C;&#x7136;&#x5F8C;&#x7528;http.DefaultClient.Do(req)&#x4F86;&#x9032;&#x884C;&#x9019;&#x7B87;http&#x8ACB;&#x6C42;&#x3002;</p>
<p>&#x7DF4;&#x7FD2;8.11:&#x7DCA;&#x63A5;&#x7740;8.4.4&#x4E2D;&#x7684;mirroredQuery&#x6D41;&#x7A0B;&#xFF0C;&#x5BE6;&#x73FE;&#x4E00;&#x7B87;&#x4F75;&#x767C;&#x8ACB;&#x6C42;url&#x7684;fetch&#x7684;&#x8B8A;&#x7A2E;&#x3002;&#x5679;&#x7B2C;&#x4E00;&#x7B87;&#x8ACB;&#x6C42;&#x8FD4;&#x8FF4;&#x6642;&#xFF0C;&#x76F4;&#x63A5;&#x53D6;&#x6D88;&#x5176;&#x5B83;&#x7684;&#x8ACB;&#x6C42;&#x3002;</p>
</section>

View File

@@ -48,7 +48,7 @@
<body>
<div class="book" data-level="8.10" data-chapter-title="示例: 聊天服務" data-filepath="ch8/ch8-10.md" data-basepath=".." data-revision="Wed Dec 09 2015 15:54:13 GMT+0800 (中国标准时间)">
<div class="book" data-level="8.10" data-chapter-title="示例: 聊天服務" data-filepath="ch8/ch8-10.md" data-basepath=".." data-revision="Mon Dec 14 2015 11:30:54 GMT+0800 (中国标准时间)">
<div class="book-summary">

View File

@@ -48,7 +48,7 @@
<body>
<div class="book" data-level="8" data-chapter-title="Goroutines和Channels" data-filepath="ch8/ch8.md" data-basepath=".." data-revision="Wed Dec 09 2015 15:54:13 GMT+0800 (中国标准时间)">
<div class="book" data-level="8" data-chapter-title="Goroutines和Channels" data-filepath="ch8/ch8.md" data-basepath=".." data-revision="Mon Dec 14 2015 11:30:54 GMT+0800 (中国标准时间)">
<div class="book-summary">